LoPy deepsleep general issues/questions
-
Hello, I need to use deepsleep to reduce the power consumption in order to send some sensor data with LoRa. I really need it because without it LoPy consumes 100mA.
So, from what I read here the deepsleep works like a reset, it will not run the code wirtten afterwards and it will repeat main.py (so i don't need to implement a while loop ?). The problems are the following:
First of all, I have this one regarding the library :
File "/flash/lib/deepsleep.py", line 32, in init
ValueError: invalid argument(s) valueAt line 32 : self.uart = UART(1, baudrate=10000, pins=(COMM_PIN, ))
It is ok after COMM_PIN to be blank space ? The lib is taken from here https://github.com/pycom/pycom-libraries/tree/master/deepsleep and I've done the latest firmware upgrade.From what I know the deepsleep can be implemented with command line like : ds.go_to_sleep( number of seconds ), if I have LoPy+deepsleep shield+Pysense can I still use it? Or shoud I try machine.sleep(number of seconds) ?(And what is the difference?)
Last one: if it runs the main.py continously why do I need to use nvram_save / nvram_restore ? The process is : read data -> send it -> sleep -> repeat.
Thank you in advance !!!
-
@johncaipa I have quite a lot of stuff going on in my code, but the main differences from yours are:
- I always to the initial join manually, so I don't try to detect whether that already happened or not.
- I use OTAA joins
- I keep the socket blocking
- I do wake on pin using the Pysense
Here's a slimmed down version. Important! Don't use this code as is! This code requires a Pysense firmware which supports INT wake-up.
from network import LoRa from network import WLAN from network import Server import socket import time import binascii from pysense import Pysense import pycom import machine app_version = '2017-10-31-01' SLEEP = 150 py = Pysense() def read_pin(): return 1 if py.peek_memory(0xE) & 0x02 else 0 pycom.heartbeat(0) app_eui = binascii.unhexlify('***') app_key = binascii.unhexlify('***') # Whatever happens, init the LoRa stack and (try to) restore its state lora = LoRa(mode=LoRa.LORAWAN, adr=True) lora.nvram_restore() # This part is not called from the code, only manually def doJoin(lora): print(time.localtime(),"LoRa join") lora.join(activation=LoRa.OTAA, auth=(app_eui, app_key), timeout=0) while not lora.has_joined(): time.sleep(1) print('Not yet joined...') lora.nvram_save() print(time.localtime(),"done LoRa join") # This is the debug mode, entered when the button is pressed def debugMode(): # set led to orange pycom.rgbled(0x402000) wlan = WLAN() wlan.init(mode=WLAN.AP, ssid='LoPy-'+binascii.hexlify(lora.mac()).decode('ascii'), auth=(WLAN.WPA2,'***'), channel=7, antenna=WLAN.INT_ANT) server = Server() # set led to green once we're ready pycom.rgbled(0x00ff00) # This is a modified version of the Pysense go_to_sleep which keeps INT working def go_to_sleep2(): PORTC_ADDR = const(0x00E) ANSELA_ADDR = const(0x18C) ANSELB_ADDR = const(0x18D) ANSELC_ADDR = const(0x18E) ADCON0_ADDR = const(0x9D) CMD_GO_SLEEP = const(0x21) # disable back-up power to the pressure sensor py.mask_bits_in_memory(PORTC_ADDR, ~(1 << 7)) py.poke_memory(ADCON0_ADDR, 0) # disable the ADC py.poke_memory(ANSELA_ADDR, ~(1 << 3)) # Don't touch RA3 so that button wake up works py.poke_memory(ANSELB_ADDR, 0xFF) py.poke_memory(ANSELC_ADDR, ~((1 << 7)|(1 << 1))) py._write(bytes([CMD_GO_SLEEP]), wait=False) # kill the run pin Pin('P3', mode=Pin.OUT, value=0) def sendData(val): s = socket.socket(socket.AF_LORA, socket.SOCK_RAW) s.setsockopt(socket.SOL_LORA, socket.SO_CONFIRMED, 0) s.send(bytes([val])); # This is the main function def doTheWork(): if py.button_pressed(): debugMode() return p = read_pin() sendData(p) # send is blocking and should have waited long enough, but I'm a belt-and-suspenders kind of guy time.sleep(2) lora.nvram_save() py.setup_sleep(SLEEP) # Set edge py.poke_memory(0x95, py.peek_memory(0x95) & ~0x40 if p else py.peek_memory(0x95) | 0x40) # Enable INTE # If your Pysense is not running a version of the firmware that supports it, this will probably lock it up py.poke_memory(0x0b, py.peek_memory(0x0b) | 0x10) go_to_sleep2() doTheWork()
Note: I haven't tried running this version of the code, so I may have cut a bit too much.
-
@jcaron can you share a functional code ?, here mine:
https://forum.pycom.io/topic/1668/has-anyone-successfully-used-lora-nvram_save-and-restore/11
-
@johncaipa that’s quite incorrect. I have devices running for weeks using the save/restore APIs, and I can guarantee you that frame counters are indeed saved.
Some additional state (channels...) wasn’t, but it was added in the latest firmware release.
Note that you should not join in each cycle, as that will indeed reset counters.
-
@jmarcelino said in LoPy deepsleep general issues/questions:
Once you enter deepsleep the RAM contents are gone, including your LoRaWAN session keys and counters. nvram_save / nvram_restore is an efficient way to store and restore these parameters without having to rejoin the LoRaWAN network.
supposedly it is like that,, but with nvram_save / nvram_restore the packet counters is not stored, it restarts in each cycle.
-
Found the answer: it will not find the module if you just run it, you need to upload it and there it will be in internal registers, same works with Pysense module.
-
Not sure if this your post was a question or a statement but just to be sure you only need the
from deepsleep import DeepSleep
line, and notimport deepsleep
-
There I have the basic example and the lib from https://github.com/pycom/pycom-libraries , I begin to feel really bad because I really miss something.
-
My basic main.py is the following:
from pysense import Pysense from LIS2HH12 import LIS2HH12 from SI7006A20 import SI7006A20 from LTR329ALS01 import LTR329ALS01 from MPL3115A2 import MPL3115A2,ALTITUDE,PRESSURE import pycom import time py = Pysense() si = SI7006A20(py) lt = LTR329ALS01(py) li = LIS2HH12(py) mp = MPL3115A2(py,mode=ALTITUDE) mpp = MPL3115A2(py,mode=PRESSURE) while True: data1=mp.temperature() data2=mpp.pressure() data3=si.humidity() data4=lt.light() treasure=['Temperature=',data1,'Pressure=',data2,'Humidity=',data3,'Light=',data4] print(treasure) py.setup_sleep(5) py.go_to_sleep()
If I don't have the while loop: it reads the data, it's not printing it and it goes to sleep JUST ONCE
If I have the while loop: it will show the data continously only if I don't use the last 2 lines, otherwise it will not display the info.
And it gets really buggy, (in the terminal normal info like Type help() for more information it displays only Type help() fo and I have to use a lot of CTRL+C and CTRL+B to restore and run it again. Is there a way to erase all content from LoPy? I think the "Upload" doesn't quite overwrite the code very well.
@jmarcelino I always use this name but I open only the folder I need when I want to run the code. I will change the name just to be shure in the future.
-
@iplooky
Do you have any other file named deepsleep.py , maybe in your /flash/ folder ?You don’t need a loop, once the board “wakes” it will run your script again.
-
@jmarcelino thank you for the answer.
Yes, the error appeared with from deepsleep import DeepSleep and ds=DeepSleep(), really don't know why.
Do I need a while loop in order to read the sensor data, send it and go to sleep or the reset effect of py.setup_sleep covers it? (I've tried both ways before putting this question but as far I can see LoPy behaves strange (again) and from experience I'll need to reset it a couple of times).
-
@iplooky
Yes, the current deepsleep via the shield or the Pysense acts as a reset and the LoPy restarts including running the boot and main.py files.First of all, I have this one regarding the library :
File "/flash/lib/deepsleep.py", line 32, in initHow are you calling it and when does this error appear?
from deepsleep import DeepSleep ds = DeepSleep()
should work fine?
From what I know the deepsleep can be implemented with command line like : ds.go_to_sleep( number of seconds ), if I have LoPy+deepsleep shield+Pysense can I still use it? Or shoud I try machine.sleep(number of seconds) ?(And what is the difference?)
If you have the Pysense you don't need the deepsleep shield, the sleep functionality is already built-in into the Pysense.
What you call depends if you are using the deepsleep shield or the Pysense
For the deepsleep shield:
# ds = DeepSleep() ds.go_to_sleep( *number of seconds* )
for the Pysense:
# py = Pysense() py.setup_sleep(*number of seconds*) py.go_to_sleep()
I'm not sure what machine.sleep is - maybe you mean time.sleep(s), which just waits a few seconds but doesn't decrease power?
There is a machine.deepsleep(ms) which uses the internal sleep modes and thus doesn't need the deepsleep shield but it had some problems on the existing LoPy. It does work correctly on the OEM modules will be the best option in future hardware.
Last one: if it runs the main.py continously why do I need to use nvram_save / nvram_restore ? The process is : read data -> send it -> sleep -> repeat.
Once you enter deepsleep the RAM contents are gone, including your LoRaWAN session keys and counters. nvram_save / nvram_restore is an efficient way to store and restore these parameters without having to rejoin the LoRaWAN network.