.. note::
¡Hola, bienvenido a la comunidad de entusiastas de SunFounder Raspberry Pi & Arduino & ESP32 en Facebook! Sumérgete más en Raspberry Pi, Arduino y ESP32 junto con otros entusiastas.
**¿Por qué unirse?**
- **Soporte experto**: Resuelve problemas postventa y desafíos técnicos con ayuda de nuestra comunidad y equipo.
- **Aprender y compartir**: Intercambia consejos y tutoriales para mejorar tus habilidades.
- **Avances exclusivos**: Obtén acceso anticipado a nuevos anuncios de productos y adelantos.
- **Descuentos especiales**: Disfruta de descuentos exclusivos en nuestros productos más recientes.
- **Promociones y sorteos festivos**: Participa en sorteos y promociones especiales en épocas festivas.
👉 ¿Listo para explorar y crear con nosotros? Haz clic en [|link_sf_facebook|] y únete hoy.
.. _3.1.5_c_mcp3008:
3.1.5 Indicador de batería (MCP3008)
=====================================
.. note::
.. image:: img/mcp3008_and_adc0834.jpg
:width: 25%
:align: left
Dependiendo de la versión de tu kit, identifica si tienes **ADC0834** o **MCP3008** y continúa con la sección correspondiente.
Introducción
--------------
En este proyecto, haremos un dispositivo indicador de batería que puede
mostrar visualmente el nivel de batería en una barra de LED (*LED Bargraph*).
.. warning::
No uses componentes de batería que superen los 3.3V para evitar sobrecargas, lo que podría dañar el chip o la Raspberry Pi.
Componentes requeridos
------------------------------
En este proyecto, necesitamos los siguientes componentes.
.. image:: img/list2_Battery_Indicator.png
:align: center
Diagrama esquemático
-----------------------------------
============ ======== ======== ===
T-Board Name physical wiringPi BCM
SPICE0 Pin 24 10 8
SPIMOSI Pin 19 12 10
SPIMISO Pin 21 13 9
SPISCLK Pin 23 14 11
GPIO25 Pin 22 6 25
GPIO12 Pin 32 26 12
GPIO16 Pin 36 27 16
GPIO20 Pin 38 28 20
GPIO21 Pin 40 29 21
GPIO5 Pin 29 21 5
GPIO6 Pin 31 22 6
GPIO13 Pin 33 23 13
GPIO19 Pin 35 24 19
GPIO26 Pin 37 25 26
============ ======== ======== ===
.. image:: img/schematic_battery_indicator_mcp3008.png
:align: center
Procedimiento experimental
---------------------------
**Paso 1:** Monta el circuito.
.. image:: img/july24_3.1.5_battery_indicator_mcp3008.png
**Para usuarios de lenguaje C**
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
**Paso 2:** Ve a la carpeta del código.
.. raw:: html
.. code-block::
cd ~/davinci-kit-for-raspberry-pi/c/3.1.5-2/
**Paso 3:** Compila el código.
.. raw:: html
.. code-block::
gcc 3.1.5_BatteryIndicator.c -lwiringPi
**Paso 4:** Ejecuta el archivo compilado.
.. raw:: html
.. code-block::
sudo ./a.out
Después de ejecutar el programa, conecta por separado el pin 3 del MCP3008 y el GND a dos cables, y luego conéctalos a los polos de una batería. Podrás ver que se enciende el LED correspondiente en la barra de LED indicando el nivel de carga (rango de medición: 0-5V).
.. note::
Si no funciona después de ejecutarlo, o aparece el error: \"wiringPi.h: No such file or directory\", revisa :ref:`install_wiringpi`.
Código
--------
.. code-block:: c
#include
#include
#include
#define SPI_CHANNEL 0
#define SPI_SPEED 1000000 // 1MHz
#define VREF 3.3
int pins[10] = {6, 26, 27, 28, 29, 21, 22, 23, 24, 25};
int read_ADC(int channel)
{
if (channel < 0 || channel > 7) return -1;
unsigned char buffer[3];
buffer[0] = 1; // Bit de inicio
buffer[1] = (8 + channel) << 4; // Modo de entrada simple
buffer[2] = 0;
wiringPiSPIDataRW(SPI_CHANNEL, buffer, 3);
int value = ((buffer[1] & 3) << 8) | buffer[2];
return value;
}
void LedBarGraph(int value) {
for (int i = 0; i < 10; i++) {
if (i < value)
digitalWrite(pins[i], HIGH);
else
digitalWrite(pins[i], LOW);
}
}
int main(void)
{
if (wiringPiSetup() == -1) {
printf("¡Error en la configuración de wiringPi!\n");
return 1;
}
if (wiringPiSPISetup(SPI_CHANNEL, SPI_SPEED) == -1) {
printf("¡Error en la configuración de SPI!\n");
return 1;
}
for (int i = 0; i < 10; i++) {
pinMode(pins[i], OUTPUT);
digitalWrite(pins[i], HIGH);
}
while (1) {
int analogVal = read_ADC(0); // MCP3008 CH0
if (analogVal < 0) continue;
float voltage = analogVal * VREF / 1023.0;
int level = analogVal * 10 / 1024;
if (level > 10) level = 10;
LedBarGraph(level);
printf("ADC Value: %d\tVoltage: %.2f V\tLevel: %d\n", analogVal, voltage, level);
delay(200);
}
return 0;
}
Explicación del código
----------------------
.. code-block:: c
int read_ADC(int channel)
{
if (channel < 0 || channel > 7) return -1;
unsigned char buffer[3];
buffer[0] = 1; // Bit de inicio
buffer[1] = (8 + channel) << 4; // Modo de entrada simple, CH0~CH7
buffer[2] = 0;
wiringPiSPIDataRW(SPI_CHANNEL, buffer, 3);
int value = ((buffer[1] & 3) << 8) | buffer[2]; // Combinar resultado de 10 bits
return value;
}
Esta función lee valores analógicos del chip MCP3008 usando SPI.
El parámetro ``channel`` selecciona una de las 8 entradas analógicas (CH0–CH7).
El MCP3008 devuelve un valor digital de 10 bits entre 0 y 1023 que representa el voltaje analógico.
.. code-block:: c
void LedBarGraph(int value) {
for (int i = 0; i < 10; i++) {
if (i < value)
digitalWrite(pins[i], HIGH); // Encender LED (cableado activo en HIGH)
else
digitalWrite(pins[i], LOW); // Apagar LED
}
}
Esta función controla una barra de LED de 10 segmentos.
Cada LED representa 1/10 del rango de voltaje.
Los LED se encienden en orden hasta el nivel especificado.
Nota: Esta versión asume que los ánodos de los LED están conectados a GPIO y los cátodos a GND (activo en HIGH).
.. code-block:: c
int main(void)
{
if (wiringPiSetup() == -1) {
printf("¡Error en la configuración de wiringPi!\n");
return 1;
}
if (wiringPiSPISetup(SPI_CHANNEL, SPI_SPEED) == -1) {
printf("¡Error en la configuración de SPI!\n");
return 1;
}
for (int i = 0; i < 10; i++) {
pinMode(pins[i], OUTPUT);
digitalWrite(pins[i], HIGH); // Inicializar todos los LED en ON
}
while (1) {
int analogVal = read_ADC(0); // Leer voltaje en CH0
if (analogVal < 0) continue;
float voltage = analogVal * VREF / 1023.0;
int level = analogVal * 10 / 1024; // Mapear a niveles 0–10
if (level > 10) level = 10;
LedBarGraph(level); // Mostrar nivel en LED
printf("ADC Value: %d\tVoltage: %.2f V\tLevel: %d\n", analogVal, voltage, level);
delay(200); // Frecuencia de actualización: 5 Hz
}
return 0;
}
Lógica principal del programa:
- Inicializa wiringPi y la comunicación SPI.
- Configura pines GPIO como salidas para controlar la barra de 10 LED.
- Lee continuamente el voltaje analógico a través de MCP3008 (CH0).
- Convierte la lectura a voltaje usando ``VREF = 3.3V``.
- Escala el voltaje a un nivel 0–10 y enciende los LED correspondientes.
- Muestra en consola el valor ADC, voltaje (en voltios) y el nivel de LED.
Esto actúa como un indicador visual de nivel de batería o voltímetro analógico.
**Para usuarios de lenguaje Python**
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
**Paso 2:** Configura la interfaz SPI e instala la biblioteca ``spidev`` (consulta :ref:`spi_configuration` para instrucciones detalladas). Si ya has hecho estos pasos, puedes omitirlos.
**Paso 3:** Ve a la carpeta del código.
.. raw:: html
.. code-block::
cd ~/davinci-kit-for-raspberry-pi/python
**Paso 4:** Ejecuta el archivo.
.. raw:: html
.. code-block::
sudo python3 3.1.5-2_BatteryIndicator.py
Después de ejecutar el programa, conecta por separado el pin 3 del ADC0834 y el GND a dos cables, y luego conéctalos a los polos de una batería. Podrás ver el LED correspondiente encendido en la barra de LED indicando el nivel de carga (rango de medición: 0-5V).
.. warning::
Si aparece el error ``RuntimeError: Cannot determine SOC peripheral base address``, revisa :ref:`faq_soc`.
Código
-------
.. code-block:: python
#!/usr/bin/env python3
import RPi.GPIO as GPIO
import spidev
import time
# Pines GPIO conectados a 10 LED, de izquierda a derecha
led_pins = [25, 12, 16, 20, 21, 5, 6, 13, 19, 26] # Numeración BCM
# Configuración GPIO
GPIO.setmode(GPIO.BCM)
for pin in led_pins:
GPIO.setup(pin, GPIO.OUT)
GPIO.output(pin, GPIO.LOW)
# Inicializar SPI
spi = spidev.SpiDev()
spi.open(0, 0) # Bus 0, CE0
spi.max_speed_hz = 1000000 # 1 MHz
# Leer valor desde un canal MCP3008
def read_adc(channel):
if channel < 0 or channel > 7:
return -1
r = spi.xfer2([1, (8 + channel) << 4, 0])
value = ((r[1] & 0x03) << 8) | r[2]
return value
# Encender barra de LED según el nivel
def led_bar_graph(level):
for i, pin in enumerate(led_pins):
if i < level:
GPIO.output(pin, GPIO.HIGH)
else:
GPIO.output(pin, GPIO.LOW)
# Bucle principal
try:
while True:
analog_val = read_adc(0) # Leer del canal 0
level = int(analog_val * 10 / 1023)
led_bar_graph(level)
print(f"ADC: {analog_val}, Nivel: {level}")
time.sleep(0.2)
except KeyboardInterrupt:
pass
finally:
for pin in led_pins:
GPIO.output(pin, GPIO.LOW)
GPIO.cleanup()
spi.close()
Explicación del código
----------------------
Este programa lee el voltaje analógico desde un MCP3008 y muestra el resultado en una barra de LED de 10 segmentos usando una Raspberry Pi (numeración BCM).
1. **Importar módulos**
- ``RPi.GPIO``: Controla los pines GPIO de la Raspberry Pi.
- ``spidev``: Se comunica con el MCP3008 mediante SPI.
- ``time``: Proporciona funciones de retardo/pausa.
.. code-block:: python
#!/usr/bin/env python3
import RPi.GPIO as GPIO
import spidev
import time
2. **Configuración de LED y GPIO**
Se define una lista con los 10 pines GPIO para el control de LED.
Se configuran como salida y se inicializan en LOW (apagados).
.. code-block:: python
# GPIO pins connected to 10 LEDs, ordered from left to right
led_pins = [25, 12, 16, 20, 21, 5, 6, 13, 19, 26] # BCM numbering
GPIO.setmode(GPIO.BCM)
for pin in led_pins:
GPIO.setup(pin, GPIO.OUT)
GPIO.output(pin, GPIO.LOW)
3. **Inicialización de SPI**
Se inicia el bus SPI 0 con chip enable CE0 y velocidad de 1 MHz.
.. code-block:: python
spi = spidev.SpiDev()
spi.open(0, 0) # Bus 0, CE0
spi.max_speed_hz = 1000000 # 1 MHz
4. **Función de lectura ADC**
Lee un valor analógico de un canal especificado (0–7) enviando un comando SPI de 3 bytes y decodificando el resultado de 10 bits.
.. code-block:: python
def read_adc(channel):
if channel < 0 or channel > 7:
return -1
r = spi.xfer2([1, (8 + channel) << 4, 0])
value = ((r[1] & 0x03) << 8) | r[2]
return value
5. **Función de barra de LED**
Enciende los LED hasta el nivel especificado; los demás permanecen apagados.
.. code-block:: python
def led_bar_graph(level):
for i, pin in enumerate(led_pins):
if i < level:
GPIO.output(pin, GPIO.HIGH)
else:
GPIO.output(pin, GPIO.LOW)
6. **Bucle principal**
Lee continuamente el valor ADC, lo escala a un nivel de 0–10 y actualiza la barra de LED.
También imprime el valor ADC y el nivel para monitoreo.
.. code-block:: python
try:
while True:
analog_val = read_adc(0)
level = int(analog_val * 10 / 1023)
led_bar_graph(level)
print(f"ADC: {analog_val}, Level: {level}")
time.sleep(0.2)
7. **Limpieza al salir**
Apaga todos los LED, libera los recursos GPIO y cierra la interfaz SPI al presionar ``Ctrl+C``.
.. code-block:: python
except KeyboardInterrupt:
pass
finally:
for pin in led_pins:
GPIO.output(pin, GPIO.LOW)
GPIO.cleanup()
spi.close()