2020-03-22 16:40:18 +01:00
|
|
|
# SPDX-License-Identifier: LGPL-3.0-or-later
|
|
|
|
# Copyright (C) 2020 Daniel Thompson
|
|
|
|
|
2020-05-14 22:36:33 +02:00
|
|
|
"""nRF-family RTC driver
|
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
"""
|
2020-02-03 20:12:04 +01:00
|
|
|
|
2020-05-09 15:18:39 +02:00
|
|
|
import machine
|
2021-06-18 22:43:47 +02:00
|
|
|
import micropython
|
2020-02-19 20:32:06 +01:00
|
|
|
import time
|
|
|
|
|
2020-02-03 20:12:04 +01:00
|
|
|
#class Stim(object):
|
|
|
|
# def __init__(self):
|
|
|
|
# self(0)
|
|
|
|
#
|
|
|
|
# def __call__(self, v):
|
|
|
|
# self.c = v
|
|
|
|
#
|
|
|
|
# def counter(self):
|
|
|
|
# return self.c
|
|
|
|
|
|
|
|
class RTC(object):
|
2020-05-14 22:36:33 +02:00
|
|
|
"""Real Time Clock based on the nRF-family low power counter.
|
|
|
|
|
|
|
|
.. automethod:: __init__
|
|
|
|
"""
|
2020-02-03 20:12:04 +01:00
|
|
|
|
|
|
|
def __init__(self, counter):
|
2020-05-14 22:36:33 +02:00
|
|
|
"""Wrap an RTCounter to provide a fully fledged Real Time Clock.
|
|
|
|
|
|
|
|
If the PNVRAM is valid then we use it to initialize the RTC otherwise
|
|
|
|
we just make something up.
|
|
|
|
|
|
|
|
:param RTCounter counter: The RTCCounter channel to adopt.
|
|
|
|
"""
|
2020-02-03 20:12:04 +01:00
|
|
|
self.counter = counter
|
2020-05-09 15:18:39 +02:00
|
|
|
|
2020-05-10 10:34:22 +02:00
|
|
|
if machine.mem32[0x200039c0] == 0x1abe11ed and \
|
|
|
|
machine.mem32[0x200039dc] == 0x10adab1e:
|
2020-05-09 15:18:39 +02:00
|
|
|
self.lastcount = self.counter.counter()
|
2020-05-10 10:34:22 +02:00
|
|
|
self.offset = machine.mem32[0x200039c4]
|
|
|
|
self._uptime = machine.mem32[0x200039c8] // 125
|
|
|
|
else:
|
|
|
|
machine.mem32[0x200039c0] = 0x1abe11ed
|
|
|
|
machine.mem32[0x200039dc] = 0x10adab1e
|
|
|
|
self._uptime = 0
|
|
|
|
self.set_localtime((2020, 3, 1, 3, 0, 0, 0, 0))
|
2020-02-03 20:12:04 +01:00
|
|
|
|
2021-06-04 11:49:33 +02:00
|
|
|
@micropython.native
|
2020-02-03 20:12:04 +01:00
|
|
|
def update(self):
|
2020-05-14 22:36:33 +02:00
|
|
|
"""Check for counter updates.
|
|
|
|
|
|
|
|
:returns: True of the wall time has changed, False otherwise.
|
|
|
|
"""
|
2020-02-03 20:12:04 +01:00
|
|
|
newcount = self.counter.counter()
|
|
|
|
split = newcount - self.lastcount
|
|
|
|
if split == 0:
|
|
|
|
return False
|
|
|
|
if split < 0:
|
|
|
|
split += (1 << 24)
|
2020-04-18 16:34:49 +02:00
|
|
|
|
|
|
|
self.lastcount += split
|
2020-02-03 20:12:04 +01:00
|
|
|
self.lastcount &= (1 << 24) - 1
|
2021-06-04 11:49:33 +02:00
|
|
|
uptime = self._uptime
|
2020-04-18 16:34:49 +02:00
|
|
|
self._uptime += split
|
2020-05-09 15:18:39 +02:00
|
|
|
machine.mem32[0x200039c8] = self._uptime * 125
|
2020-02-03 20:12:04 +01:00
|
|
|
|
2021-06-04 11:49:33 +02:00
|
|
|
# Has the seconds count changed
|
|
|
|
return bool((self._uptime ^ uptime) & 0x08)
|
2020-02-03 20:12:04 +01:00
|
|
|
|
2020-02-19 20:32:06 +01:00
|
|
|
def set_localtime(self, t):
|
2020-05-14 22:36:33 +02:00
|
|
|
"""Set the current wall time.
|
|
|
|
|
|
|
|
:param sequence t:
|
|
|
|
Wall time formatted as (yyyy, mm, dd, HH, MM, SS), any
|
|
|
|
additional elements in sequence will be ignored.
|
|
|
|
"""
|
2020-02-19 20:32:06 +01:00
|
|
|
self.lastcount = self.counter.counter()
|
2020-02-03 20:12:04 +01:00
|
|
|
|
2020-02-19 20:32:06 +01:00
|
|
|
if len(t) < 8:
|
|
|
|
yyyy = t[0]
|
|
|
|
mm = t[1]
|
|
|
|
dd = t[2]
|
|
|
|
HH = t[3]
|
|
|
|
MM = t[4]
|
|
|
|
SS = t[5]
|
2020-02-03 20:12:04 +01:00
|
|
|
|
2020-02-19 20:32:06 +01:00
|
|
|
t = (yyyy, mm, dd, HH, MM, SS, 0, 0)
|
2020-02-03 20:12:04 +01:00
|
|
|
|
2020-02-19 20:32:06 +01:00
|
|
|
lt = time.mktime(t)
|
2020-04-26 15:59:29 +02:00
|
|
|
self.offset = lt - (self._uptime >> 3)
|
2020-05-09 15:18:39 +02:00
|
|
|
machine.mem32[0x200039c4] = self.offset
|
2020-02-03 20:12:04 +01:00
|
|
|
|
2020-02-19 20:32:06 +01:00
|
|
|
def get_localtime(self):
|
2020-05-14 22:36:33 +02:00
|
|
|
"""Get the current time and date.
|
|
|
|
|
|
|
|
:returns: Wall time formatted as (yyyy, mm, dd, HH, MM, SS, wday, yday)
|
|
|
|
"""
|
2020-02-03 20:12:04 +01:00
|
|
|
self.update()
|
2020-04-18 16:34:49 +02:00
|
|
|
return time.localtime(self.offset + (self._uptime >> 3))
|
2020-02-19 20:32:06 +01:00
|
|
|
|
|
|
|
def get_time(self):
|
2020-05-14 22:36:33 +02:00
|
|
|
"""Get the current time.
|
|
|
|
|
|
|
|
:returns: Wall time formatted as (HH, MM, SS)
|
|
|
|
"""
|
2020-02-19 20:32:06 +01:00
|
|
|
localtime = self.get_localtime()
|
|
|
|
return localtime[3:6]
|
2020-03-07 12:49:35 +01:00
|
|
|
|
2021-06-04 11:49:33 +02:00
|
|
|
@micropython.native
|
2020-11-14 13:24:28 +01:00
|
|
|
def time(self):
|
|
|
|
"""Get time in the same format as time.time"""
|
|
|
|
return self.offset + (self._uptime >> 3)
|
|
|
|
|
2020-04-18 16:34:49 +02:00
|
|
|
@property
|
|
|
|
def uptime(self):
|
|
|
|
"""Provide the current uptime in seconds."""
|
|
|
|
return self._uptime // 8
|
|
|
|
|
2020-03-07 12:49:35 +01:00
|
|
|
def get_uptime_ms(self):
|
|
|
|
"""Return the current uptime in milliseconds."""
|
2020-04-18 16:34:49 +02:00
|
|
|
return self._uptime * 125
|