.. note::
Ciao, benvenuto nella comunità SunFounder Raspberry Pi & Arduino & ESP32 Enthusiasts su Facebook! Approfondisci Raspberry Pi, Arduino ed ESP32 insieme ad altri appassionati.
**Perché unirsi?**
- **Supporto esperto**: Risolvi problemi post-vendita e sfide tecniche con l’aiuto della nostra comunità e del nostro team.
- **Impara e condividi**: Scambia suggerimenti e tutorial per migliorare le tue competenze.
- **Anteprime esclusive**: Ottieni accesso anticipato agli annunci di nuovi prodotti e anteprime.
- **Sconti speciali**: Goditi sconti esclusivi sui nostri prodotti più recenti.
- **Promozioni e giveaway festivi**: Partecipa a giveaway e promozioni durante le festività.
👉 Pronto a esplorare e creare con noi? Clicca [|link_sf_facebook|] e unisciti oggi stesso!
.. _3.1.5_c_mcp3008:
3.1.5 Indicatore di Batteria (MCP3008)
======================================
.. note::
.. image:: img/mcp3008_and_adc0834.jpg
:width: 25%
:align: left
A seconda della versione del kit, verifica se possiedi **ADC0834** o **MCP3008** e procedi con la sezione corrispondente.
Introduzione
------------
In questo progetto, realizzeremo un dispositivo indicatore di batteria che può mostrare visivamente il livello della batteria sul LED Bargraph.
.. warning::
Non utilizzare componenti della batteria che superino i 3,3 V per evitare sovraccarichi che potrebbero danneggiare il chip o il Raspberry Pi.
Componenti necessari
--------------------
Per questo progetto, abbiamo bisogno dei seguenti componenti.
.. image:: img/list2_Battery_Indicator.png
:align: center
Schema elettrico
----------------
============ ======== ======== ===
T-Board Name fisico 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
Procedure sperimentali
----------------------
**Passo 1:** Montare il circuito.
.. image:: img/july24_3.1.5_battery_indicator_mcp3008.png
**Per utenti linguaggio C**
^^^^^^^^^^^^^^^^^^^^^^^^^^^
**Passo 2:** Accedere alla cartella del codice.
.. raw:: html
.. code-block::
cd ~/davinci-kit-for-raspberry-pi/c/3.1.5-2/
**Passo 3:** Compilare il codice.
.. raw:: html
.. code-block::
gcc 3.1.5_BatteryIndicator.c -lwiringPi
**Passo 4:** Eseguire il file eseguibile.
.. raw:: html
.. code-block::
sudo ./a.out
Dopo l'esecuzione del programma, collega separatamente un filo dal 3° pin di MCP3008 e un filo dal GND ai due poli della batteria. Potrai vedere il LED corrispondente sul LED Bargraph accendersi per mostrare il livello della batteria (intervallo di misurazione: 0-5V).
.. note::
Se non funziona dopo l'esecuzione, o compare l'errore: "wiringPi.h: No such file or directory", fare riferimento a :ref:`install_wiringpi`.
Codice
------
.. 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; // Start bit
buffer[1] = (8 + channel) << 4; // Single-ended mode
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("setup wiringPi failed!\n");
return 1;
}
if (wiringPiSPISetup(SPI_CHANNEL, SPI_SPEED) == -1) {
printf("SPI setup failed!\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;
}
Spiegazione del codice
----------------------
.. code-block:: c
int read_ADC(int channel)
{
if (channel < 0 || channel > 7) return -1;
unsigned char buffer[3];
buffer[0] = 1; // Start bit
buffer[1] = (8 + channel) << 4; // Modalità single-ended, CH0~CH7
buffer[2] = 0;
wiringPiSPIDataRW(SPI_CHANNEL, buffer, 3);
int value = ((buffer[1] & 3) << 8) | buffer[2]; // Combina il risultato a 10 bit
return value;
}
Questa funzione legge valori analogici dal chip MCP3008 tramite SPI.
Il parametro ``channel`` seleziona uno degli 8 ingressi analogici (CH0–CH7).
L’MCP3008 restituisce un valore digitale a 10 bit compreso tra 0 e 1023, che rappresenta la tensione analogica.
.. code-block:: c
void LedBarGraph(int value) {
for (int i = 0; i < 10; i++) {
if (i < value)
digitalWrite(pins[i], HIGH); // Accende il LED (cablaggio attivo HIGH)
else
digitalWrite(pins[i], LOW); // Spegne il LED
}
}
Questa funzione controlla una barra LED a 10 segmenti.
Ogni LED rappresenta 1/10 dell’intervallo di tensione.
I LED vengono accesi in ordine fino al livello specificato.
Nota: Questa versione presuppone che gli anodi dei LED siano collegati ai GPIO e i catodi a GND (attivo HIGH).
.. code-block:: c
int main(void)
{
if (wiringPiSetup() == -1) {
printf("setup wiringPi failed!\n");
return 1;
}
if (wiringPiSPISetup(SPI_CHANNEL, SPI_SPEED) == -1) {
printf("SPI setup failed!\n");
return 1;
}
for (int i = 0; i < 10; i++) {
pinMode(pins[i], OUTPUT);
digitalWrite(pins[i], HIGH); // Inizializza tutti i LED su ON
}
while (1) {
int analogVal = read_ADC(0); // Legge la tensione su CH0
if (analogVal < 0) continue;
float voltage = analogVal * VREF / 1023.0;
int level = analogVal * 10 / 1024; // Mappa a livelli 0–10
if (level > 10) level = 10;
LedBarGraph(level); // Mostra il livello sui LED
printf("ADC Value: %d\tVoltage: %.2f V\tLevel: %d\n", analogVal, voltage, level);
delay(200); // Frequenza di aggiornamento: 5 Hz
}
return 0;
}
Logica principale del programma:
- Inizializza wiringPi e la comunicazione SPI.
- Imposta i pin GPIO come uscite per controllare la barra LED a 10 segmenti.
- Legge continuamente la tensione analogica tramite MCP3008 (CH0).
- Converte la lettura in tensione usando ``VREF = 3.3V``.
- Scala la tensione su un livello da 0 a 10 e accende i LED corrispondenti.
- Mostra il valore ADC grezzo, la tensione (in volt) e il livello LED sulla console seriale.
Questo progetto funge da indicatore visivo del livello di batteria o da voltmetro analogico.
**Per utenti linguaggio Python**
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
**Passo 2:** Configurare l'interfaccia SPI e installare la libreria ``spidev`` (vedere :ref:`spi_configuration` per le istruzioni dettagliate). Se hai già completato questi passaggi, puoi saltarli.
**Passo 3:** Accedere alla cartella del codice.
.. raw:: html
.. code-block::
cd ~/davinci-kit-for-raspberry-pi/python
**Passo 4:** Eseguire il file.
.. raw:: html
.. code-block::
sudo python3 3.1.5-2_BatteryIndicator.py
Dopo l'esecuzione del programma, collega separatamente un filo dal 3° pin di ADC0834 e un filo dal GND ai due poli della batteria. Potrai vedere il LED corrispondente sul LED Bargraph accendersi per mostrare il livello della batteria (intervallo di misurazione: 0-5V).
.. warning::
Se appare l'errore ``RuntimeError: Cannot determine SOC peripheral base address``, fare riferimento a :ref:`faq_soc`
**Codice**
.. note::
Puoi **Modificare/Reimpostare/Copiare/Eseguire/Interrompere** il codice qui sotto. Ma prima, devi andare nel percorso del codice sorgente come ``davinci-kit-for-raspberry-pi/python``. Dopo aver modificato il codice, puoi eseguirlo direttamente per vedere l'effetto.
.. raw:: html
.. code-block:: python
#!/usr/bin/env python3
import RPi.GPIO as GPIO
import spidev
import time
# 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 setup
GPIO.setmode(GPIO.BCM)
for pin in led_pins:
GPIO.setup(pin, GPIO.OUT)
GPIO.output(pin, GPIO.LOW)
# Initialize SPI
spi = spidev.SpiDev()
spi.open(0, 0) # Bus 0, CE0
spi.max_speed_hz = 1000000 # 1 MHz
# Read value from MCP3008 channel
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
# Light up LED bar graph according to value
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)
# Main loop
try:
while True:
analog_val = read_adc(0) # Read from MCP3008 channel 0
level = int(analog_val * 10 / 1023)
led_bar_graph(level)
print(f"ADC: {analog_val}, Level: {level}")
time.sleep(0.2)
except KeyboardInterrupt:
pass
finally:
for pin in led_pins:
GPIO.output(pin, GPIO.LOW)
GPIO.cleanup()
spi.close()
Spiegazione del codice
----------------------
Questo programma legge la tensione analogica da un MCP3008 ADC e mostra il risultato su un LED Bargraph a 10 LED utilizzando un Raspberry Pi (layout pin BCM).
1. **Importazione moduli**
- ``RPi.GPIO`` controlla i pin GPIO del Raspberry Pi.
- ``spidev`` comunica con MCP3008 tramite SPI.
- ``time`` fornisce funzionalità di ritardo/pausa.
.. code-block:: python
#!/usr/bin/env python3
import RPi.GPIO as GPIO
import spidev
import time
2. **Configurazione GPIO per i LED**
Viene definita una lista di 10 pin GPIO per il controllo dei LED. Questi pin sono configurati come uscita e inizializzati a LOW (spenti).
.. 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. **Inizializzazione SPI**
Inizializza il bus SPI 0 e il chip enable 0 (CE0) per comunicare con MCP3008.
La velocità di comunicazione è impostata a 1 MHz.
.. code-block:: python
spi = spidev.SpiDev()
spi.open(0, 0) # Bus 0, CE0
spi.max_speed_hz = 1000000 # 1 MHz
4. **Funzione di lettura ADC**
Legge un valore analogico da un canale specificato di MCP3008 (0–7). La funzione invia un comando SPI di 3 byte e decodifica il risultato a 10 bit.
.. 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. **Funzione LED Bar Graph**
Accende i LED in base al livello analogico. Se il livello è 7, i primi 7 LED saranno accesi e i restanti spenti.
.. 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. **Ciclo principale**
Legge continuamente l'ingresso analogico dal canale 0, scala il risultato a un valore da 0 a 10 e aggiorna il display LED di conseguenza. Stampa i valori ADC e di livello per il monitoraggio.
.. 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. **Pulizia all'uscita**
Quando si preme ``Ctrl+C``, il programma spegne tutti i LED, ripristina lo stato dei GPIO e chiude l'interfaccia SPI.
.. code-block:: python
except KeyboardInterrupt:
pass
finally:
for pin in led_pins:
GPIO.output(pin, GPIO.LOW)
GPIO.cleanup()
spi.close()