Bewegingsdetector
Infraroodsensoren zijn sensoren die gebruik maken van infrarood licht. Ze worden voor verschillende doeleinden gebruikt, zoals het detecteren van beweging, ontvangers voor IrDA-systemen, contactloze temperatuursensoren en het maken van afstandschattingen. Beeldvormende infraroodsensoren worden bovendien ingezet als nachtzichtapparatuur.
Veelgebruikt wordt de passieve infraroodsensor (PIR) voor bewegingsdetectie. Je hebt ze wellicht al gezien, ze worden gebruikt in alarmsystemen (de sensoren in de hoeken van kamers zijn meestal PIR sensors). Alle voorwerpen die een temperatuur heeft van meer dan het absolute nulpunt zendt infrarood licht uit. Infrarood licht is niet zichtbaar voor het menselijk oog, maar kan wel door elektronica gedetecteerd worden.
Een sensor is passief als deze geen licht uitstraalt om beweging te detecteren. Het past zich aan de infrarood straling van een kamer aan en reageert op veranderingen daarin. Elk bewegend voorwerp zal deze infrarood straling beïnvloeden en dus door de PIR opgemerkt worden.
We houden ons niet bezig met de interne werking van een PIR sensor, wel zijn we geïnteresseerd in hoe we een PIR sensor op de Raspberry Pi kunnen aansluiten.
PIR sensor aansluiten
Voor je de Raspberry Pi opstart, verbind je de PIR sensor met de Pi. Let daarbij goed op de aansluitingen, niet op elke PIR staan de aansluitingen op dezelfde plaats en/of volgorde.
Met behulp van drie female-to-female draadverbindingen verbind je de PIR sensor (SR501) met de Raspberry Pi:
- Verbindt de VCC pen van de PIR sensor met de 5V pen van de Raspberry Pi.
- Verbindt de GND pen van de PIR sensor met de massa van de Raspberry Pi. De PIR sensor kan nu via de Raspberry Pi van stroom worden voorzien.
- Verbindt de OUT pen (gele kabel op de afbeelding) van de PIR sensor met GPIO 23 van de Raspberry Pi. De spanning op deze pen verandert wanneer de PIR sensor beweging detecteert en geeft dit zo door aan de Raspberry Pi.
De PIR bewegingssensor testen
Zorg dat Raspberry Pi OS bijgewerkt is.
We gaan een script gebruiken om wat code te schrijven waarbij we tekst laten verschijnen bij het detecteren van beweging.
Start Geany (Menu > Programmeren > Geany), maak een nieuw Bash script (Bestand > Opslaan als met de bestandsnaam beweging.sh
)
en typ de volgende code:
#!/bin/bash
# Eerst definiëren we enkele functies
# Functie waarmee je wacht op beweging:
wait_for_motion () {
while :
do
pir=$(pinctrl get 23)
if [[ $pir == *"hi"* ]]; then
break
fi
sleep 1
done
}
# Functie waarmee je wacht tot er geen beweging meer is
wait_for_no_motion () {
while :
do
pir=$(pinctrl get 23)
if [[ $pir == *"lo"* ]]; then
break
fi
sleep 1
done
}
# Hier begint ons programma
# Initialiseer GPIO 23 als input met standaard laag (pull down)
pinctrl set 23 ip pd
# Oneindig lang worden bewegingen gemeld
# Breek het programma af met Ctrl+c
while :
do
wait_for_motion
echo "Beweging waargenomen"
wait_for_no_motion
echo "Geen beweging"
done
Hoe is het script opgebouwd? Eerst schrijven we functies om het script te laten wachten op beweging of juist geen beweging. De PIR sensor meldt beweging op GPIO 23. Dus zorgen we dat GPIO 23 de status van de PIR sensor kan inlezen (invoer of input) en zetten deze standaard op laag (low of massa, geen beweging).
pi@raspberrypi:~ $ pinctrl set 23 ip pd
De status van GPIO 23 kan je met de volgende opdracht opvragen:
pi@raspberrypi:~ $ pinctrl get 23
23: ip pd | lo // GPIO23 = input
Dit was zonder dat de PIR beweging detecteerde, met beweging krijgen we:
pi@raspberrypi:~ $ pinctrl get 23
23: ip pd | hi // GPIO23 = input
M.a.w. de PIR-uitgang krijgt bij beweging de waarde hi
, zonder beweging de waarde lo
.
Dit moet constant gebeuren en gebruiken daarvoor een while
lus.
In de lus controleren we GPIO pen 23 op beweging ( hi
).
Als dit zo is ([ $pir == *"hi"* ]
) wordt de lus verlaten (break
).
Als dit niet zo is, wordt 1 seconde gewacht (sleep 1
) en wordt de lus opnieuw uitgevoerd.
M.a.w. de functie is pas klaar als er beweging (eerste functie) of juist geen beweging (tweede functie) werd waargenomen.
De sleep
opdracht zorgt dat ons script 1 seconden niets doet, waardoor andere programma's en het besturingssysteem andere taken kunnen uitvoeren.
Zonder de sleep
opdracht zou alle processorkracht naar ons script gaan en de Raspberry Pi vastlopen.
Ook ons programma bestaat uit een eindeloze lus.
Daarin wachten we met behulp van onze functies respectievelijk op beweging en geen beweging waarbij we dit telkens melden (echo
).
De programmalus onderbreken we manueel met Ctrl+c en het script wordt beëindigd.
Sla het bestand op en voer in een terminal de volgende opdracht uit om het script te kunnen uitvoeren:
pi@raspberrypi:~ $ chmod +x beweging.sh
Activeer Geany terug en druk F5 om het programma te starten.
Telkens wanneer de PIR sensor beweging detecteert verschijnt de tekst Beweging waargenomen! in het terminal venster. Indien er geen beweging meer wordt opgemerkt, verschijnt de tekst Geen beweging!. Druk Ctrl+c om het programma af te sluiten.
Op de SR501 PIR sensor staan twee oranje componenten die je met een Philips schroevendraaier kunt aanpassen. Dit zijn potentiometers en ze laten toe de gevoeligheid en de detectievertraging van de sensor in te stellen. Je begint meestal met een maximale gevoeligheid en de kortste detectievertraging. De gevoeligheid regelt de afstand waarop de sensor nog reageert (van 3 tot 7 meter, in wijzerzin draaien om te vergroten). De detectie vertraging regel je tussen de 0,3 en 600 seconden (in wijzerzin draaien om te vergroten)
Praktijk opdracht
- Schakel de Raspberry Pi uit en verbindt de Pi Camera met de Raspberry Pi.
- Start de Raspberry Pi en zorg dat de Pi Camera werkt. M.a.w. installeer de rpicam opdrachten en neem een foto.
- Nu kunnen we de Pi Camera inzetten om bij het detecteren van beweging een opname te maken.
We beginnen met het weergeven van een voorvertoning bij een bewegingsdetectie.
De functies blijven onveranderd aanwezig.
Binnen de
while
lus starten we een voorvertoning bij elke bewegingsdetectie, m.a.w. pas het volgende programma-onderdeel aan:# Oneindig lang worden bewegingen gemeld
# Breek het programma af met Ctrl+c
timeout=4
preview="0,50,640,360"
while :
do
wait_for_motion
rpicam-hello -t $((timeout*1000)) -p $preview
wait_for_no_motion
echo "Geen beweging"
done - Met wat meer code kunnen we een opname maken om later te bekijken.
Zo wil je bij meerdere indringers in de kamer, ze allemaal vastleggen en niet alleen de laatste.
Om dit te realiseren, moeten we voor elke opname een andere bestandsnaam hebben.
De eenvoudigste en veiligste is een naam te maken die bestaat uit de datum en tijd van de opname.
Een voorbeeld: de bestandsnaam 2024-07-30_16.37.42.h264 wordt gebruikt voor een opname gemaakt op 30 juli 2024 om 16u37m42s. Dit gebruikt het formaatYEAR-MONTH-DAY_HOUR.MINUTE.SECOND.h264
dat ook door YouTube wordt gebruikt.
Daarvoor gebruiken we de dateopdracht en schrijven we code om een bestandsnaam met een datum en tijd te genereren. Daarna kan je de opname starten met de aangemaakte bestandsnaam. Juist zoals dat ging met het starten van de voorvertoning:# Oneindig lang worden bewegingen gemeld
Test het programma en laat een paar opnames maken.
# Breek het programma af met Ctrl+c
timeout=4
preview="0,50,640,360"
while :
do
wait_for_motion
bestandsnaam=$(date +"%Y-%m-%d_%H.%M.%S.h264")
rpicam-vid -t $((timeout*1000)) -p $preview -o $bestandsnaam
wait_for_no_motion
echo "Geen beweging"
done - Test de opnames in de terminal met de opdracht
ffplay <bestandsnaam>
, een voorbeeld:pi@raspberrypi:~ $
ffplay -autoexit 2024-07-30_16.37.42.h264
ffplay version 5.1.5-0+rpt1+deb12u1 Copyright (c) 2003-2024 the FFmpeg developers built with gcc 12 (Debian 12.2.0-14) configuration: --prefix=/usr --extra-version=0+rpt1+deb12u1 --toolchain=hardened --incdir=/usr/include/aarch64-linux-gnu --enable-gpl --disable-stripping --disable-mmal --enable-gnutls --enable-ladspa --enable-libaom --enable-libass --enable-libbluray --enable-libbs2b --enable-libcaca --enable-libcdio --enable-libcodec2 --enable-libdav1d --enable-libflite --enable-libfontconfig --enable-libfreetype --enable-libfribidi --enable-libglslang --enable-libgme --enable-libgsm --enable-libjack --enable-libmp3lame --enable-libmysofa --enable-libopenjpeg --enable-libopenmpt --enable-libopus --enable-libpulse --enable-librabbitmq --enable-librist --enable-librubberband --enable-libshine --enable-libsnappy --enable-libsoxr --enable-libspeex --enable-libsrt --enable-libssh --enable-libsvtav1 --enable-libtheora --enable-libtwolame --enable-libvidstab --enable-libvorbis --enable-libvpx --enable-libwebp --enable-libx265 --enable-libxml2 --enable-libxvid --enable-libzimg --enable-libzmq --enable-libzvbi --enable-lv2 --enable-omx --enable-openal --enable-opencl --enable-opengl --enable-sand --enable-sdl2 --disable-sndio --enable-libjxl --enable-neon --enable-v4l2-request --enable-libudev --enable-epoxy --libdir=/usr/lib/aarch64-linux-gnu --arch=arm64 --enable-pocketsphinx --enable-librsvg --enable-libdc1394 --enable-libdrm --enable-vout-drm --enable-libiec61883 --enable-chromaprint --enable-frei0r --enable-libx264 --enable-libplacebo --enable-librav1e --enable-shared libavutil 57. 28.100 / 57. 28.100 libavcodec 59. 37.100 / 59. 37.100 libavformat 59. 27.100 / 59. 27.100 libavdevice 59. 7.100 / 59. 7.100 libavfilter 8. 44.100 / 8. 44.100 libswscale 6. 7.100 / 6. 7.100 libswresample 4. 7.100 / 4. 7.100 libpostproc 56. 6.100 / 56. 6.100 Input #0, h264, from '2024-07-30_16.37.42.h264':B sq= 0B f=0/0 Duration: N/A, bitrate: N/A Stream #0:0: Video: h264 (High), yuv420p(tv, smpte170m/smpte170m/bt709, progressive), 640x480, 1000k tbr, 1200k tbn nan M-V: nan fd= 60 aq= 0KB vq= 0KB sq= 0B f=0/0 - Soms verloopt de opname niet zo onopvallend.
Soms wordt bij elke opname een rode led op de Pi Camera actief, dit valt enorm op.
Deze led kan je uitschakelen in het configuratiebestand /boot/firmware/config.txt van de Raspberry Pi.
Open als systeembeheerder dit configuratiebestand
pi@raspberrypi:~ $
en voeg de volgende regel toe:sudo nano /boot/firmware/config.txt
disable_camera_led=1
Sla de aanpassing op en herstart de Pi om de nieuwe instelling te activeren. Test of alles nog werkt. - Voor een bepaalde toepassing bestaan soms meerdere oplossingen. Je moet dus voor je aan iets begint, nadenken over hoe je het gaat aanpakken. Hou rekening met jouw eigen vaardigheden, zo zal een handige Harry eerder voor een hardware oplossing kiezen, een programmeur (sterk in logisch denken) kiest eerder voor een software oplossing. Ga ook op zoek naar reeds bestaande oplossingen, de meeste bestaande oplossingen, zeker als die open source zijn, kan je verder uitbouwen om te voldoen aan jouw eigen noden. We gaan het project om beweging te detecteren helemaal omgooien. We gebruiken minder hardware en vervangen die door software. Schakel de Raspberry Pi daarvoor eerst uit en koppel de PIR sensor los.
- Start de Raspberry Pi opnieuw op.
- We gaan de Pi Camera inzetten als bewegingsdetector.
Het gebruik van de Pi Camera heeft zowel voordelen als nadelen:
Nadelen:
- Je moet de videobeelden afkomstig van de Pi Camera analyseren om beweging te detecteren. De bewegingsdetectie wordt dus verzorgt door software en belast de processor. Een PIR belast de processor niet.
- De Pi Camera werkt niet in het donker. Een PIR werkt met infrarood en dus ook in het donker. Met een NoIR Pi Camera en infraroodverlichting werkt het wel.
Voordelen:
- Je kunt de bewegingsdetectie eenvoudiger instellen. Zo kan je het te controleren gebied als video of foto bekijken. De gevoeligheid en andere parameters kan je via de software instellen en niet op de PIR zelf. Belangrijk als de bewegingsdetector op een moeilijk te bereiken plaats staat.
- Je kunt de Pi Camera ook gebruiken om opnames te maken. M.a.w. je hebt dan maar één hardware-uitbreiding nodig, nml. de Pi Camera.
- Om dit te realiseren, gebruiken we postProcessing (zie Pi Camera).
We gaan ook elke beweging opnemen en dus niet langer wachten tot er geen beweging meer is.
M.a.w. de functie
wait_for_no_motion
wordt overbodig. Het nieuwe script wordt dan:#!/bin/bash
# Eerst definiëren we enkele instellingen (constanten)
timeout=4
preview="0,50,640,360"
motion=$(cat << EOF
{
"motion_detect" :
{
"roi_x" : 0.1,
"roi_y" : 0.1,
"roi_width" : 0.8,
"roi_height" : 0.8,
"difference_m" : 0.1,
"difference_c" : 10,
"region_threshold" : 0.005,
"frame_period" : 5,
"hskip" : 2,
"vskip" : 2,
"verbose" : 1
}
}
EOF
)
echo $motion > postProcess.json
# Daarna definiëren we enkele functies
# Functie waarmee je wacht op beweging:
wait_for_motion () {
while read -r line ; do
if [[ "$line" == *"Motion detected"* ]]; then
# Stop de bewegingsdetectie om de camera te gebruiken om een video te maken
pkill rpicam-hello
echo "Er is beweging."
break
fi
# We gebruiken --nopreview om zo onopvallend mogelijk beweging te detecteren.
# Gebruik een preview om het detectiegebied te bekijken/af te regelen.
done < <(rpicam-hello -t 0 --nopreview --lores-width 128 --lores-height 96 --post-process-file postProcess.json 2>&1)
}
# Hier begint ons programma
# Oneindig lang worden bewegingen gemeld
# Breek het programma af met Ctrl+c
while :
do
wait_for_motion
bestandsnaam=$(date +"%Y-%m-%d_%H.%M.%S.h264")
rpicam-vid -t $((timeout*1000)) -p $preview -o $bestandsnaam
done - Test of het script werkt.
- Als je een specialist wat betreft camera bewegingsdetectie wilt gebruiken, kijk dan eens naar Motion.