Sommario:
2025 Autore: John Day | [email protected]. Ultima modifica: 2025-01-23 14:50
Preambolo
Questo Instructable descrive in dettaglio come creare un'implementazione non bloccante del sensore di gesti APDS9960 utilizzando SparkFun_APDS-9960_Sensor_Arduino_Library.
introduzione
Quindi probabilmente ti starai chiedendo cos'è il non bloccante? O addirittura bloccando per quella materia?
Ancora più importante, perché è importante avere qualcosa di non bloccante giusto?
Ok, quindi quando un microprocessore esegue un programma esegue in sequenza righe di codice e nel farlo effettua chiamate e ritorna dalle funzioni secondo l'ordine in cui le hai scritte.
Una chiamata di blocco è solo una chiamata a qualsiasi tipo di funzionalità che provoca l'interruzione dell'esecuzione, ovvero una chiamata di funzione in cui il chiamante non riprenderà l'esecuzione fino a quando la funzione chiamata non avrà terminato l'esecuzione.
Quindi perchè è importante?
Nel caso in cui tu abbia scritto del codice che deve eseguire regolarmente molte funzioni in sequenza come leggere una temperatura, leggere un pulsante e aggiornare un display, se il codice per aggiornare il display fosse una chiamata di blocco, il tuo sistema non risponderà a pressione dei pulsanti e variazioni di temperatura, poiché il processore trascorrerà tutto il tempo in attesa che il display si aggiorni e non leggerà lo stato del pulsante o l'ultima temperatura.
Da parte mia, voglio creare un dispositivo desktop IoT compatibile con MQTT su WiFi che legga i valori di temperatura/umidità locali e remoti, i livelli di luce ambientale, la pressione barometrica, tenga traccia del tempo, visualizzi tutti questi parametri su un LCD, acceda a un uSD scheda in tempo reale, leggere gli ingressi dei pulsanti, scrivere sui LED di uscita e monitorare i gesti per controllare le cose nella mia infrastruttura IoT e tutto ciò che deve essere controllato da un ESP8266-12.
Sfortunatamente le uniche due fonti della libreria APDS9960 che ho trovato sono state le librerie SparkFun e AdaFruit, entrambe estratte dal codice dell'applicazione da Avago (il produttore di ADPS9960) e possiedono una chiamata chiamata "readGesture" che contiene a while(1){}; loop che, se utilizzato nel progetto precedente, provoca il ripristino dell'ESP8266-12E ogni volta che il sensore ADPS9960 si satura (ad es. quando un oggetto è rimasto nelle immediate vicinanze o c'era un'altra sorgente IR che illuminava il sensore).
Di conseguenza, per risolvere questo comportamento ho scelto di spostare l'elaborazione dei gesti su un secondo processore per cui ESP8266-12E è diventato il microcontrollore master e questo sistema lo slave, come illustrato rispettivamente nelle immagini 1 e 2 sopra, nei diagrammi Panoramica del sistema e Composizione del sistema. La figura 3 mostra il circuito prototipo.
Per limitare le modifiche che dovevo apportare al mio codice esistente, ho anche scritto una classe/libreria wrapper dal nome fantasioso "APDS9960_NonBlocking".
Quella che segue è una spiegazione dettagliata della soluzione non bloccante.
Di quali parti ho bisogno?
Se vuoi costruire la soluzione I2C che funziona con la libreria APDS9960_NonBlocking avrai bisogno delle seguenti parti.
- 1 sconto ATMega328P qui
- 1 sconto PCF8574P qui
- 6 resistori da 10K qui
- 4 off 1K resistori qui
- 1 diodo 1N914 qui
- 1 off PN2222 NPN Transistor qui
- 1 fuori da 16 MHz di cristallo qui
- 2 condensatori da 0.1uF qui
- 1 condensatore elettrolitico da 1000uF qui
- 1 condensatore elettrolitico da 10uF qui
- 2 condensatori da 22 pF qui
Se si desidera leggere l'uscita del sensore gestuale tramite l'interfaccia parallela, è possibile eliminare il PCF8574P e tre resistori da 10K.
Di che software ho bisogno?
Arduino IDE 1.6.9
Di quali competenze ho bisogno?
Per configurare il sistema, usa il codice sorgente (fornito) e crea i circuiti necessari di cui avrai bisogno quanto segue;
- Una conoscenza minima dell'elettronica,
- Conoscenza di Arduino e del suo IDE,
- Una comprensione di come programmare un Arduino incorporato (vedi Instructable "Programmazione di ATTiny85, ATTiny84 e ATMega328P: Arduino come ISP")
- Un po' di pazienza.
Argomenti trattati
- Breve panoramica del circuito
- Breve panoramica del software
- Test del dispositivo di rilevamento dei gesti APDS9960
- Conclusione
- Riferimenti
Passaggio 1: panoramica del circuito
Il circuito è diviso in due sezioni;
- Il primo è la conversione seriale da I2C a parallelo realizzata tramite i resistori R8…10 e IC1. Qui R8…R10 imposta l'indirizzo I2C per il chip di espansione I/O a 8 bit IC1 e NXP PCF8574A. Gli intervalli di indirizzi validi per questo dispositivo sono rispettivamente 0x38 … 0x3F. Nell'esempio del software I2C fornito 'I2C_APDS9960_TEST.ino' '#define GESTURE_SENSOR_I2C_ADDRESS' dovrebbe essere modificato per adattarsi a questo intervallo di indirizzi.
-
Tutti gli altri componenti formano un Arduino Uno slave embedded e hanno le seguenti funzioni;
- R1, T1, R2 e D1 forniscono un ingresso di reset del dispositivo slave. Qui un impulso alto attivo su IC1 - P7 forzerà il reset di U1.
- R3, R4, sono resistori di limitazione di corrente per le linee TX/RX di programmazione del dispositivo integrato.
- C5 e R7 consentono all'IDE Arduino di programmare automaticamente U1 tramite un impulso sulla linea DTR di un dispositivo FTDI collegato.
- R5 e R6 sono resistori pull-up I2C per APDS9960 con C6 che fornisce il disaccoppiamento della linea di alimentazione locale.
- U1, C1, C2 e Q1 formano rispettivamente l'Arduino Uno integrato e il suo orologio.
- Infine C3 e C4 forniscono il disaccoppiamento della linea di alimentazione locale per U1.
Passaggio 2: panoramica del software
Preambolo
Per compilare correttamente questo codice sorgente avrai bisogno delle seguenti librerie extra per programmare l'Arduino Uno U1 integrato;
SparkFun_APDS9960.h
- Di: Steve Quinn
- Scopo: questa è una versione biforcuta del sensore SparkFun APDS9960 biforcato da jonn26/SparkFun_APDS-9960_Sensor_Arduino_Library. Ha alcune modifiche per aiutare con il debug e ha un rilevatore desensibilizzato per ridurre l'attivazione spuria.
- Da:
APDS9960_NonBlocking.h
- Di: Steve Quinn
- Scopo: fornisce un'interfaccia pulita per incorporare questa implementazione non bloccante del sensore di gesti APDS9960 nel codice Arduino.
- Da:
Vedere il seguente Instructable su come programmare un microcontrollore Arduino Uno (ATMega328P) integrato se non si ha familiarità con come ottenere ciò;
PROGRAMMAZIONE DI ATTINY85, ATTINY84 E ATMEGA328P: ARDUINO COME ISP
Panoramica funzionale
Il microcontrollore slave incorporato ATMega328P interroga la linea INT dall'ADPS9960. Quando questa linea diventa bassa, il microcontrollore legge i registri ADPS9960 e determina se è stato rilevato un gesto valido. Se è stato rilevato un gesto valido, il codice per questo gesto 0x0…0x6, 0xF viene posizionato sulla porta B e 'nGestureAvailable' viene dichiarato basso.
Quando il dispositivo Master vede 'nGestureAvailable' attivo, legge il valore sulla porta B, quindi emette temporaneamente un segnale basso di 'nGestureClear' per confermare la ricezione dei dati.
Il dispositivo slave quindi de-asserisce 'nGestureAvailable' alto e cancella i dati sulla porta B. La figura 5 sopra mostra una schermata presa da un analizzatore di stati logici durante un ciclo completo di rilevamento/lettura.
Panoramica del codice
La figura 1 sopra descrive come funziona il software in U1 lo slave Arduino Uno incorporato, insieme alla figura 2 come interagiscono le due attività in background / in primo piano. La figura 3 è un segmento di codice che illustra come utilizzare APDS9960_NonBlockinglibrary. La figura 4 fornisce una mappatura tra i pin digitali Arduino Uno e i pin hardware effettivi sull'ATMega328P.
Dopo il ripristino, il microcontrollore slave incorporato inizializza l'APDS9960 consentendo al rilevamento dei gesti di attivare la sua uscita INT e configura il suo I/O, collegando la routine di servizio di interrupt (ISR) 'GESTURE_CLEAR()' al vettore di interruzione INT0 (pin 2 digitale, pin 4 hardware IC)., configurandolo per un trigger sul fronte di discesa. Questo costituisce l'input nGestureClear dal dispositivo master.
Il pin di uscita Interrupt 'INT' dall'APDS9960 è collegato al Pin digitale 4, Hardware IC Pin 6 che è configurato come ingresso a U1.
La linea di segnale 'nGestureAvailable' sul pin 7 digitale, pin 13 hardware IC è configurata come uscita e impostata su alta, inattiva (de-asserta).
Infine, i bit 0…3 della porta B rispettivamente sono configurati come uscite e impostati a livello basso. Questi formano il nibble di dati che rappresenta i vari tipi di gesto rilevati; Nessuno = 0x0, Errore = 0xF, Su = 0x1, Giù = 0x2, Sinistra = 0x3, Destra = 0x4, Vicino = 0x5 e Lontano = 0x6.
L'attività in background 'Loop' è pianificata che interroga continuamente l'Interrupt output INT di APDS9960 tramite la lettura del Pin digitale 4. Quando l'output INT dell'APDS9960 diventa attivo basso indicando che il sensore è stato attivato, il microcontrollore tenta di interpretare qualsiasi gesto chiamando 'readGesture()' con mentre (1) {}; ciclo infinito.
Se è stato rilevato un gesto valido, questo valore viene scritto sulla porta B, viene asserito l'output 'nGestureAvailable' e viene impostato il semaforo booleano 'bGestureAvailable', impedendo la registrazione di ulteriori movimenti.
Una volta che il master rileva l'uscita 'nGestureAvailable' attiva, legge questo nuovo valore e emette un segnale 'nGestureClear' attivo basso. Questo fronte di discesa attiva la pianificazione dell'attività in primo piano 'ISR GESTURE_CLEAR()', sospendendo l'esecuzione dell'attività in background 'Loop', cancellando la porta B, il semaforo 'bGestureAvailable' e l'output 'nGestureAvailable'.
L'attività in primo piano 'GESTURE_CLEAR()' è ora sospesa e l'attività in background 'Loop' è stata riprogrammata. È ora possibile rilevare ulteriori gesti dall'APDS9960.
Utilizzando in questo modo attività in primo piano/in background attivate da interrupt, il potenziale ciclo infinito in 'readGesture()' del dispositivo slave non influenzerà il funzionamento del dispositivo master e non ostacolerà nemmeno l'esecuzione del dispositivo slave. Ciò costituisce la base di un sistema operativo in tempo reale molto semplice (RTOS).
Nota: il prefisso "n" significa attivo basso o asserito come in "nGestureAvailable"
Passaggio 3: test del dispositivo di rilevamento dei gesti non bloccante APDS9960
Preambolo
Anche se il modulo APDS9960 è fornito con +5v, utilizza un regolatore +3v3 a bordo, il che significa che le sue linee I2C sono compatibili con +3v3 e non +5v. Questo è il motivo per cui ho scelto di utilizzare Arduino Due conforme a +3v3 come microcontrollore di prova, per ovviare alla necessità di traslatori di livello.
Se, tuttavia, desideri utilizzare un vero Arduino Uno, dovrai spostare le linee I2C su U1. Vedere il seguente Instructable in cui ho allegato un utile set di diapositive (I2C_LCD_With_Arduino) che fornisce molti suggerimenti pratici sull'utilizzo di I2C.
Test dell'interfaccia I2C
Le immagini 1 e 2 sopra mostrano come impostare e programmare il sistema per l'interfaccia I2C. Dovrai prima scaricare e installare la libreria APDS9960_NonBlocking. qui
Test dell'interfaccia parallela
Le immagini 3 e 4 mostrano lo stesso dettaglio per l'interfaccia parallela
Passaggio 4: conclusione
Generale
Il codice funziona bene e rileva i gesti in modo reattivo senza falsi positivi. È stato installato e funzionante per alcune settimane come dispositivo slave nel mio prossimo progetto. Ho provato molte diverse modalità di errore (e così ha fatto il curioso gatto domestico Quinn) che in precedenza aveva provocato un reset di ESP8266-12, senza alcun effetto negativo.
Possibili miglioramenti
-
L'ovvio. Riscrivi la libreria del sensore di gesti APDS9960 in modo che non sia bloccante.
In realtà ho contattato Broadcom che mi ha messo in contatto con un distributore locale che ha prontamente ignorato la mia richiesta di supporto, ma non sono uno SparkFun o AdaFruit, immagino. Quindi questo probabilmente dovrà aspettare un po'
- Porta il codice su un microcontrollore slave più piccolo. L'utilizzo di un ATMega328P per un'attività è un po' eccessivo. Anche se inizialmente ho guardato ATTiny84, ho smesso di usarne uno perché sentivo che la dimensione compilata del codice era una linea di confine. Con l'ulteriore sovraccarico di dover modificare la libreria APDS9960 per lavorare con una diversa libreria I2C.
Passaggio 5: riferimenti
Necessario per programmare l'arduino integrato (ATMega328P - U1)
SparkFun_APDS9960.h
- Di: Steve Quinn
- Scopo: questa è una versione biforcuta del sensore SparkFun APDS9960 biforcato da jonn26/SparkFun_APDS-9960_Sensor_Arduino_Library. Ha alcune modifiche per aiutare con il debug e ha un rilevatore desensibilizzato per ridurre l'attivazione spuria.
- Da:
Necessario per incorporare questa funzionalità non bloccante nel codice Arduino e fornire esempi funzionati
APDS9960_NonBlocking.h
- Di: Steve Quinn
- Scopo: fornisce un'interfaccia pulita per incorporare questa implementazione non bloccante del sensore di gesti APDS9960 nel codice Arduino.
- Da:
Sistema operativo in tempo reale
https://en.wikipedia.org/wiki/Sistema_operativo_in tempo reale
Scheda tecnica APDS9960
https://cdn.sparkfun.com/assets/learn_tutorials/3/2/1/Avago-APDS-9960-datasheet.pdf
Consigliato:
Misurazione della distanza di prossimità con il sensore di gesti APDS9960: 6 passaggi
Misurazione della distanza di prossimità con sensore gestuale APDS9960: In questo tutorial impareremo come misurare la distanza utilizzando un sensore gestuale APDS9960, arduino e Visuino.Guarda il video
Come utilizzare il sensore di gesti APDS9960 con Arduino: 7 passaggi
Come utilizzare il sensore di gesti APDS9960 con Arduino: in questo tutorial impareremo come utilizzare il sensore di gesti APDS9960 con Arduino per visualizzare le indicazioni delle mani sul display OLED utilizzando il software Visuino. Guarda il video
Implementazione dell'hardware TicTacToe utilizzando RaspberryPi: 4 passaggi
Implementazione hardware TicTacToe utilizzando RaspberryPi: questo progetto mira a costruire un modello TicTacToe interattivo utilizzando due LED di colore diverso che denotano i due giocatori utilizzando un Raspberry Pi. L'idea qui era di implementarla su una scala più grande in un vicolo - immagina una griglia di semi-globi 3x3 (li
Implementazione LiFi, Uso Sencillo: 5 passaggi
Implementación LiFi, Uso Sencillo: La transmisión de datos por vía de luz (LiFi) è un problema reale. Per risolvere il problema si tratta di una prima approssimazione, se desarrolló un dispositivo può tenere una comunicazione in una via per medio raggio, un congiunto di LED a infrarossi
Controllo di un anello LED Neopixel con un sensore di gesti: 3 passaggi (con immagini)
Controllo di un anello LED Neopixel con un sensore di gesti: in questo tutorial giocheremo con un sensore di gesti (APDS-9960) e un anello neopixel per imparare a combinarli entrambi utilizzando un Arduino UNO. Il prodotto finale risponderà a gesti sinistro - destro animando il movimento led a destra oa sinistra, e per u