2020-07-18 10:05:46 +02:00
|
|
|
/****************************************************************************
|
|
|
|
* arch/arm/src/nrf52/nrf52_rtc.c
|
|
|
|
*
|
|
|
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
|
|
|
* contributor license agreements. See the NOTICE file distributed with
|
|
|
|
* this work for additional information regarding copyright ownership. The
|
|
|
|
* ASF licenses this file to you under the Apache License, Version 2.0 (the
|
|
|
|
* "License"); you may not use this file except in compliance with the
|
|
|
|
* License. You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
|
|
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
|
|
* License for the specific language governing permissions and limitations
|
|
|
|
* under the License.
|
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Included Files
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#include <nuttx/config.h>
|
|
|
|
|
2021-05-18 08:59:14 +02:00
|
|
|
#include <assert.h>
|
2020-07-18 10:05:46 +02:00
|
|
|
#include <debug.h>
|
|
|
|
|
|
|
|
#include <nuttx/arch.h>
|
|
|
|
#include <nuttx/irq.h>
|
|
|
|
|
2022-03-11 17:41:15 +01:00
|
|
|
#include "arm_internal.h"
|
2020-07-18 10:05:46 +02:00
|
|
|
#include "hardware/nrf52_rtc.h"
|
|
|
|
|
|
|
|
#include "nrf52_rtc.h"
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Pre-processor Definitions
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Private Types
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
struct nrf52_rtc_priv_s
|
|
|
|
{
|
2022-04-17 08:01:48 +02:00
|
|
|
struct nrf52_rtc_ops_s *ops;
|
|
|
|
uint32_t base;
|
|
|
|
uint32_t irq;
|
|
|
|
uint8_t chan;
|
|
|
|
bool inuse;
|
2020-07-18 10:05:46 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Private Function Prototypes
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/* RTC registers access *****************************************************/
|
|
|
|
|
2022-04-17 08:01:48 +02:00
|
|
|
static uint32_t nrf52_rtc_getreg(struct nrf52_rtc_dev_s *dev,
|
2020-07-18 10:05:46 +02:00
|
|
|
uint32_t offset);
|
2022-04-17 08:01:48 +02:00
|
|
|
static void nrf52_rtc_putreg(struct nrf52_rtc_dev_s *dev,
|
2020-07-18 10:05:46 +02:00
|
|
|
uint32_t offset,
|
|
|
|
uint32_t value);
|
|
|
|
|
|
|
|
/* RTC helpers **************************************************************/
|
|
|
|
|
2022-04-17 08:01:48 +02:00
|
|
|
static uint32_t nrf52_rtc_irq2reg(struct nrf52_rtc_dev_s *dev,
|
2020-07-18 10:05:46 +02:00
|
|
|
uint8_t s);
|
2022-04-17 08:01:48 +02:00
|
|
|
static uint32_t nrf52_rtc_evt2reg(struct nrf52_rtc_dev_s *dev,
|
2020-07-25 03:59:05 +02:00
|
|
|
uint8_t evt);
|
2020-07-18 10:05:46 +02:00
|
|
|
|
|
|
|
/* RTC operations ***********************************************************/
|
|
|
|
|
2022-04-17 08:01:48 +02:00
|
|
|
static int nrf52_rtc_start(struct nrf52_rtc_dev_s *dev);
|
|
|
|
static int nrf52_rtc_stop(struct nrf52_rtc_dev_s *dev);
|
|
|
|
static int nrf52_rtc_clear(struct nrf52_rtc_dev_s *dev);
|
|
|
|
static int nrf52_rtc_trgovrflw(struct nrf52_rtc_dev_s *dev);
|
|
|
|
static int nrf52_rtc_getcounter(struct nrf52_rtc_dev_s *dev,
|
|
|
|
uint32_t *cc);
|
|
|
|
static int nrf52_rtc_setcc(struct nrf52_rtc_dev_s *dev, uint8_t i,
|
2020-07-18 10:05:46 +02:00
|
|
|
uint32_t cc);
|
2022-04-17 08:01:48 +02:00
|
|
|
static int nrf52_rtc_getcc(struct nrf52_rtc_dev_s *dev, uint8_t i,
|
|
|
|
uint32_t *cc);
|
|
|
|
static int nrf52_rtc_setpre(struct nrf52_rtc_dev_s *dev, uint16_t pre);
|
|
|
|
static int nrf52_rtc_setisr(struct nrf52_rtc_dev_s *dev, xcpt_t handler,
|
|
|
|
void * arg);
|
|
|
|
static int nrf52_rtc_enableint(struct nrf52_rtc_dev_s *dev, uint8_t s);
|
|
|
|
static int nrf52_rtc_disableint(struct nrf52_rtc_dev_s *dev, uint8_t s);
|
|
|
|
static int nrf52_rtc_checkint(struct nrf52_rtc_dev_s *dev, uint8_t s);
|
|
|
|
static int nrf52_rtc_ackint(struct nrf52_rtc_dev_s *dev, uint8_t s);
|
|
|
|
static int nrf52_rtc_enableevt(struct nrf52_rtc_dev_s *dev, uint8_t evt);
|
|
|
|
static int nrf52_rtc_disableevt(struct nrf52_rtc_dev_s *dev,
|
2020-08-24 15:43:34 +02:00
|
|
|
uint8_t evt);
|
2022-04-17 08:01:48 +02:00
|
|
|
static uint32_t nrf52_rtc_getbase(struct nrf52_rtc_dev_s *dev);
|
2020-07-18 10:05:46 +02:00
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Private Data
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/* NRF52 RTC ops */
|
|
|
|
|
|
|
|
struct nrf52_rtc_ops_s nrf52_rtc_ops =
|
|
|
|
{
|
|
|
|
.start = nrf52_rtc_start,
|
|
|
|
.stop = nrf52_rtc_stop,
|
|
|
|
.clear = nrf52_rtc_clear,
|
|
|
|
.trgovrflw = nrf52_rtc_trgovrflw,
|
2020-09-06 04:27:29 +02:00
|
|
|
.getcounter = nrf52_rtc_getcounter,
|
2020-07-18 10:05:46 +02:00
|
|
|
.setcc = nrf52_rtc_setcc,
|
|
|
|
.getcc = nrf52_rtc_getcc,
|
|
|
|
.setpre = nrf52_rtc_setpre,
|
|
|
|
.setisr = nrf52_rtc_setisr,
|
|
|
|
.enableint = nrf52_rtc_enableint,
|
|
|
|
.disableint = nrf52_rtc_disableint,
|
|
|
|
.checkint = nrf52_rtc_checkint,
|
2020-07-25 03:59:05 +02:00
|
|
|
.ackint = nrf52_rtc_ackint,
|
|
|
|
.enableevt = nrf52_rtc_enableevt,
|
|
|
|
.disableevt = nrf52_rtc_disableevt,
|
2021-01-30 21:31:30 +01:00
|
|
|
.getbase = nrf52_rtc_getbase,
|
2020-07-18 10:05:46 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
#ifdef CONFIG_NRF52_RTC0
|
|
|
|
/* RTC0 */
|
|
|
|
|
|
|
|
struct nrf52_rtc_priv_s g_nrf52_rtc0_priv =
|
|
|
|
{
|
|
|
|
.ops = &nrf52_rtc_ops,
|
|
|
|
.base = NRF52_RTC0_BASE,
|
|
|
|
.irq = NRF52_IRQ_RTC0,
|
|
|
|
.chan = 3,
|
|
|
|
.inuse = false,
|
|
|
|
};
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef CONFIG_NRF52_RTC1
|
|
|
|
/* RTC1 */
|
|
|
|
|
|
|
|
struct nrf52_rtc_priv_s g_nrf52_rtc1_priv =
|
|
|
|
{
|
|
|
|
.ops = &nrf52_rtc_ops,
|
|
|
|
.base = NRF52_RTC1_BASE,
|
|
|
|
.irq = NRF52_IRQ_RTC1,
|
|
|
|
.chan = 4,
|
|
|
|
.inuse = false,
|
|
|
|
};
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef CONFIG_NRF52_RTC2
|
|
|
|
/* RTC2 */
|
|
|
|
|
|
|
|
struct nrf52_rtc_priv_s g_nrf52_rtc2_priv =
|
|
|
|
{
|
|
|
|
.ops = &nrf52_rtc_ops,
|
|
|
|
.base = NRF52_RTC2_BASE,
|
|
|
|
.irq = NRF52_IRQ_RTC2,
|
|
|
|
.chan = 4,
|
|
|
|
.inuse = false,
|
|
|
|
};
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Private Functions
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: nrf52_rtc_getreg
|
|
|
|
*
|
|
|
|
* Description:
|
|
|
|
* Get a 32-bit register value by offset
|
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
2022-04-17 08:01:48 +02:00
|
|
|
static uint32_t nrf52_rtc_getreg(struct nrf52_rtc_dev_s *dev,
|
2020-07-18 10:05:46 +02:00
|
|
|
uint32_t offset)
|
|
|
|
{
|
|
|
|
DEBUGASSERT(dev);
|
|
|
|
|
|
|
|
return getreg32(((struct nrf52_rtc_priv_s *)dev)->base + offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: nrf52_rtc_putreg
|
|
|
|
*
|
|
|
|
* Description:
|
|
|
|
* Put a 32-bit register value by offset
|
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
2022-04-17 08:01:48 +02:00
|
|
|
static void nrf52_rtc_putreg(struct nrf52_rtc_dev_s *dev,
|
2020-07-18 10:05:46 +02:00
|
|
|
uint32_t offset,
|
|
|
|
uint32_t value)
|
|
|
|
{
|
|
|
|
DEBUGASSERT(dev);
|
|
|
|
|
|
|
|
putreg32(value, ((struct nrf52_rtc_priv_s *)dev)->base + offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: nrf52_rtc_irq2reg
|
|
|
|
*
|
|
|
|
* Description:
|
2020-07-25 03:59:05 +02:00
|
|
|
* Get the value of the interrupt register corresponding to the given
|
2020-07-18 10:05:46 +02:00
|
|
|
* interrupt source
|
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
2022-04-17 08:01:48 +02:00
|
|
|
static uint32_t nrf52_rtc_irq2reg(struct nrf52_rtc_dev_s *dev, uint8_t s)
|
2020-07-18 10:05:46 +02:00
|
|
|
{
|
|
|
|
uint32_t regval = 0;
|
|
|
|
|
|
|
|
switch (s)
|
|
|
|
{
|
2020-08-24 19:19:41 +02:00
|
|
|
case NRF52_RTC_EVT_TICK:
|
2020-07-18 10:05:46 +02:00
|
|
|
{
|
|
|
|
regval = RTC_INT_TICK;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-08-24 19:19:41 +02:00
|
|
|
case NRF52_RTC_EVT_OVRFLW:
|
2020-07-18 10:05:46 +02:00
|
|
|
{
|
|
|
|
regval = RTC_INT_OVRFLW;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-08-24 19:19:41 +02:00
|
|
|
case NRF52_RTC_EVT_COMPARE0:
|
2020-07-18 10:05:46 +02:00
|
|
|
{
|
|
|
|
regval = RTC_INT_COMPARE(0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-08-24 19:19:41 +02:00
|
|
|
case NRF52_RTC_EVT_COMPARE1:
|
2020-07-18 10:05:46 +02:00
|
|
|
{
|
|
|
|
regval = RTC_INT_COMPARE(1);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-08-24 19:19:41 +02:00
|
|
|
case NRF52_RTC_EVT_COMPARE2:
|
2020-07-18 10:05:46 +02:00
|
|
|
{
|
|
|
|
regval = RTC_INT_COMPARE(2);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-08-24 19:19:41 +02:00
|
|
|
case NRF52_RTC_EVT_COMPARE3:
|
2020-07-18 10:05:46 +02:00
|
|
|
{
|
|
|
|
regval = RTC_INT_COMPARE(3);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
rtcerr("ERROR: unsupported IRQ source %d\n", s);
|
|
|
|
regval = 0;
|
|
|
|
goto errout;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
errout:
|
|
|
|
return regval;
|
|
|
|
}
|
|
|
|
|
2020-07-25 03:59:05 +02:00
|
|
|
/****************************************************************************
|
|
|
|
* Name: nrf52_rtc_evt2reg
|
|
|
|
*
|
|
|
|
* Description:
|
|
|
|
* Get the offset of the event register corresponding to the given event
|
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
2022-04-17 08:01:48 +02:00
|
|
|
static uint32_t nrf52_rtc_evt2reg(struct nrf52_rtc_dev_s *dev,
|
2020-07-25 03:59:05 +02:00
|
|
|
uint8_t evt)
|
|
|
|
{
|
|
|
|
uint32_t regval;
|
|
|
|
|
|
|
|
switch (evt)
|
|
|
|
{
|
|
|
|
case NRF52_RTC_EVT_TICK:
|
|
|
|
{
|
|
|
|
regval = RTC_EVTEN_TICK;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case NRF52_RTC_EVT_OVRFLW:
|
|
|
|
{
|
|
|
|
regval = RTC_EVTEN_OVRFLW;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case NRF52_RTC_EVT_COMPARE0:
|
|
|
|
{
|
|
|
|
regval = RTC_EVTEN_COMPARE(0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case NRF52_RTC_EVT_COMPARE1:
|
|
|
|
{
|
|
|
|
regval = RTC_EVTEN_COMPARE(1);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case NRF52_RTC_EVT_COMPARE2:
|
|
|
|
{
|
|
|
|
regval = RTC_EVTEN_COMPARE(2);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case NRF52_RTC_EVT_COMPARE3:
|
|
|
|
{
|
|
|
|
regval = RTC_EVTEN_COMPARE(3);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
rtcerr("ERROR: unsupported EVENT %d\n", evt);
|
|
|
|
regval = 0;
|
|
|
|
goto errout;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
errout:
|
|
|
|
return regval;
|
|
|
|
}
|
|
|
|
|
2020-07-18 10:05:46 +02:00
|
|
|
/****************************************************************************
|
|
|
|
* Name: nrf52_rtc_start
|
|
|
|
****************************************************************************/
|
|
|
|
|
2022-04-17 08:01:48 +02:00
|
|
|
static int nrf52_rtc_start(struct nrf52_rtc_dev_s *dev)
|
2020-07-18 10:05:46 +02:00
|
|
|
{
|
|
|
|
DEBUGASSERT(dev);
|
|
|
|
|
|
|
|
nrf52_rtc_putreg(dev, NRF52_RTC_TASKS_START_OFFSET, RTC_TASKS_START);
|
|
|
|
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: nrf52_rtc_stop
|
|
|
|
****************************************************************************/
|
|
|
|
|
2022-04-17 08:01:48 +02:00
|
|
|
static int nrf52_rtc_stop(struct nrf52_rtc_dev_s *dev)
|
2020-07-18 10:05:46 +02:00
|
|
|
{
|
|
|
|
DEBUGASSERT(dev);
|
|
|
|
|
|
|
|
nrf52_rtc_putreg(dev, NRF52_RTC_TASKS_STOP_OFFSET, RTC_TASKS_STOP);
|
|
|
|
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: nrf52_rtc_clear
|
|
|
|
****************************************************************************/
|
|
|
|
|
2022-04-17 08:01:48 +02:00
|
|
|
static int nrf52_rtc_clear(struct nrf52_rtc_dev_s *dev)
|
2020-07-18 10:05:46 +02:00
|
|
|
{
|
|
|
|
DEBUGASSERT(dev);
|
|
|
|
|
|
|
|
nrf52_rtc_putreg(dev, NRF52_RTC_TASKS_CLEAR_OFFSET, RTC_TASKS_CLEAR);
|
|
|
|
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: nrf52_rtc_trgovrflw
|
|
|
|
****************************************************************************/
|
|
|
|
|
2022-04-17 08:01:48 +02:00
|
|
|
static int nrf52_rtc_trgovrflw(struct nrf52_rtc_dev_s *dev)
|
2020-07-18 10:05:46 +02:00
|
|
|
{
|
|
|
|
DEBUGASSERT(dev);
|
|
|
|
|
|
|
|
nrf52_rtc_putreg(dev, NRF52_RTC_TASKS_TRIGOVRFLW_OFFSET,
|
|
|
|
RTC_TASKS_TRIGOVRFLW);
|
|
|
|
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
2022-04-17 08:01:48 +02:00
|
|
|
static int nrf52_rtc_getcounter(struct nrf52_rtc_dev_s *dev,
|
|
|
|
uint32_t *ctr)
|
2020-09-06 04:27:29 +02:00
|
|
|
{
|
|
|
|
DEBUGASSERT(dev);
|
|
|
|
DEBUGASSERT(ctr);
|
|
|
|
|
|
|
|
*ctr = nrf52_rtc_getreg(dev, NRF52_RTC_COUNTER_OFFSET);
|
|
|
|
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
2020-07-18 10:05:46 +02:00
|
|
|
/****************************************************************************
|
|
|
|
* Name: nrf52_rtc_setcc
|
|
|
|
****************************************************************************/
|
|
|
|
|
2022-04-17 08:01:48 +02:00
|
|
|
static int nrf52_rtc_setcc(struct nrf52_rtc_dev_s *dev, uint8_t i,
|
2020-07-18 10:05:46 +02:00
|
|
|
uint32_t cc)
|
|
|
|
{
|
2022-04-17 08:01:48 +02:00
|
|
|
struct nrf52_rtc_priv_s *rtc = NULL;
|
2020-07-18 10:05:46 +02:00
|
|
|
int ret = OK;
|
|
|
|
|
|
|
|
DEBUGASSERT(dev);
|
|
|
|
|
2022-04-17 08:01:48 +02:00
|
|
|
rtc = (struct nrf52_rtc_priv_s *)dev;
|
2020-07-18 10:05:46 +02:00
|
|
|
|
|
|
|
/* Is the channel supported? */
|
|
|
|
|
|
|
|
if (i > rtc->chan)
|
|
|
|
{
|
|
|
|
rtcerr("ERROR: unsupported RTCER channel %d\n", i);
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto errout;
|
|
|
|
}
|
|
|
|
|
|
|
|
nrf52_rtc_putreg(dev, NRF52_RTC_CC_OFFSET(i), cc);
|
|
|
|
|
|
|
|
errout:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: nrf52_rtc_getcc
|
|
|
|
****************************************************************************/
|
|
|
|
|
2022-04-17 08:01:48 +02:00
|
|
|
static int nrf52_rtc_getcc(struct nrf52_rtc_dev_s *dev, uint8_t i,
|
|
|
|
uint32_t *cc)
|
2020-07-18 10:05:46 +02:00
|
|
|
{
|
2022-04-17 08:01:48 +02:00
|
|
|
struct nrf52_rtc_priv_s *rtc = NULL;
|
2020-07-18 10:05:46 +02:00
|
|
|
int ret = OK;
|
|
|
|
|
|
|
|
DEBUGASSERT(dev);
|
|
|
|
DEBUGASSERT(cc);
|
|
|
|
|
2022-04-17 08:01:48 +02:00
|
|
|
rtc = (struct nrf52_rtc_priv_s *)dev;
|
2020-07-18 10:05:46 +02:00
|
|
|
|
|
|
|
/* Is the channel supported? */
|
|
|
|
|
|
|
|
if (i > rtc->chan)
|
|
|
|
{
|
|
|
|
rtcerr("ERROR: unsupported RTCER channel %d\n", i);
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto errout;
|
|
|
|
}
|
|
|
|
|
|
|
|
*cc = nrf52_rtc_getreg(dev, NRF52_RTC_CC_OFFSET(i));
|
|
|
|
|
|
|
|
errout:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: nrf52_rtc_setpre
|
|
|
|
****************************************************************************/
|
|
|
|
|
2022-04-17 08:01:48 +02:00
|
|
|
static int nrf52_rtc_setpre(struct nrf52_rtc_dev_s *dev, uint16_t pre)
|
2020-07-18 10:05:46 +02:00
|
|
|
{
|
|
|
|
int ret = OK;
|
|
|
|
|
|
|
|
DEBUGASSERT(dev);
|
|
|
|
|
|
|
|
if (pre > RTC_PRESCALER_MAX)
|
|
|
|
{
|
|
|
|
rtcerr("ERROR: unsupported RTC prescaler %d\n", pre);
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto errout;
|
|
|
|
}
|
|
|
|
|
|
|
|
nrf52_rtc_putreg(dev, NRF52_RTC_PRESCALER_OFFSET, pre);
|
|
|
|
|
|
|
|
errout:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: nrf52_rtc_setisr
|
|
|
|
****************************************************************************/
|
|
|
|
|
2022-04-17 08:01:48 +02:00
|
|
|
static int nrf52_rtc_setisr(struct nrf52_rtc_dev_s *dev, xcpt_t handler,
|
|
|
|
void *arg)
|
2020-07-18 10:05:46 +02:00
|
|
|
{
|
2022-04-17 08:01:48 +02:00
|
|
|
struct nrf52_rtc_priv_s *rtc = NULL;
|
2020-07-18 10:05:46 +02:00
|
|
|
int ret = OK;
|
|
|
|
|
|
|
|
DEBUGASSERT(dev);
|
|
|
|
|
2022-04-17 08:01:48 +02:00
|
|
|
rtc = (struct nrf52_rtc_priv_s *)dev;
|
2020-07-18 10:05:46 +02:00
|
|
|
|
|
|
|
/* Disable interrupt when callback is removed */
|
|
|
|
|
|
|
|
if (!handler)
|
|
|
|
{
|
|
|
|
up_disable_irq(rtc->irq);
|
|
|
|
irq_detach(rtc->irq);
|
|
|
|
ret = OK;
|
|
|
|
goto errout;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Otherwise set callback and enable interrupt */
|
|
|
|
|
|
|
|
irq_attach(rtc->irq, handler, arg);
|
|
|
|
up_enable_irq(rtc->irq);
|
|
|
|
|
|
|
|
errout:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: nrf52_rtc_enableint
|
|
|
|
****************************************************************************/
|
|
|
|
|
2022-04-17 08:01:48 +02:00
|
|
|
static int nrf52_rtc_enableint(struct nrf52_rtc_dev_s *dev, uint8_t s)
|
2020-07-18 10:05:46 +02:00
|
|
|
{
|
|
|
|
uint32_t regval = 0;
|
|
|
|
int ret = OK;
|
|
|
|
|
|
|
|
DEBUGASSERT(dev);
|
|
|
|
|
|
|
|
/* Get register value for given interrupt source */
|
|
|
|
|
|
|
|
regval = nrf52_rtc_irq2reg(dev, s);
|
|
|
|
if (regval == 0)
|
|
|
|
{
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto errout;
|
|
|
|
}
|
|
|
|
|
|
|
|
nrf52_rtc_putreg(dev, NRF52_RTC_INTENSET_OFFSET, regval);
|
|
|
|
|
|
|
|
errout:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: nrf52_rtc_disableint
|
|
|
|
****************************************************************************/
|
|
|
|
|
2022-04-17 08:01:48 +02:00
|
|
|
static int nrf52_rtc_disableint(struct nrf52_rtc_dev_s *dev, uint8_t s)
|
2020-07-18 10:05:46 +02:00
|
|
|
{
|
|
|
|
uint32_t regval = 0;
|
|
|
|
int ret = OK;
|
|
|
|
|
|
|
|
DEBUGASSERT(dev);
|
|
|
|
|
|
|
|
/* Get register value for given interrupt source */
|
|
|
|
|
|
|
|
regval = nrf52_rtc_irq2reg(dev, s);
|
|
|
|
if (regval == 0)
|
|
|
|
{
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto errout;
|
|
|
|
}
|
|
|
|
|
|
|
|
nrf52_rtc_putreg(dev, NRF52_RTC_INTENCLR_OFFSET, regval);
|
|
|
|
|
|
|
|
errout:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: nrf52_rtc_checkint
|
|
|
|
****************************************************************************/
|
|
|
|
|
2022-04-17 08:01:48 +02:00
|
|
|
static int nrf52_rtc_checkint(struct nrf52_rtc_dev_s *dev, uint8_t s)
|
2020-07-18 10:05:46 +02:00
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
DEBUGASSERT(dev);
|
|
|
|
|
|
|
|
switch (s)
|
|
|
|
{
|
2020-08-24 19:19:41 +02:00
|
|
|
case NRF52_RTC_EVT_TICK:
|
2020-07-18 10:05:46 +02:00
|
|
|
{
|
|
|
|
ret = nrf52_rtc_getreg(dev, NRF52_RTC_EVENTS_TICK_OFFSET);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-08-24 19:19:41 +02:00
|
|
|
case NRF52_RTC_EVT_OVRFLW:
|
2020-07-18 10:05:46 +02:00
|
|
|
{
|
|
|
|
ret = nrf52_rtc_getreg(dev, NRF52_RTC_EVENTS_OVRFLW_OFFSET);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-08-24 19:19:41 +02:00
|
|
|
case NRF52_RTC_EVT_COMPARE0:
|
2020-07-18 10:05:46 +02:00
|
|
|
{
|
|
|
|
ret = nrf52_rtc_getreg(dev, NRF52_RTC_EVENTS_COMPARE_OFFSET(0));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-08-24 19:19:41 +02:00
|
|
|
case NRF52_RTC_EVT_COMPARE1:
|
2020-07-18 10:05:46 +02:00
|
|
|
{
|
2020-08-22 22:15:04 +02:00
|
|
|
ret = nrf52_rtc_getreg(dev, NRF52_RTC_EVENTS_COMPARE_OFFSET(1));
|
2020-07-18 10:05:46 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-08-24 19:19:41 +02:00
|
|
|
case NRF52_RTC_EVT_COMPARE2:
|
2020-07-18 10:05:46 +02:00
|
|
|
{
|
|
|
|
ret = nrf52_rtc_getreg(dev, NRF52_RTC_EVENTS_COMPARE_OFFSET(2));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-08-24 19:19:41 +02:00
|
|
|
case NRF52_RTC_EVT_COMPARE3:
|
2020-07-18 10:05:46 +02:00
|
|
|
{
|
|
|
|
ret = nrf52_rtc_getreg(dev, NRF52_RTC_EVENTS_COMPARE_OFFSET(3));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
rtcerr("ERROR: unsupported IRQ source %d\n", s);
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto errout;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
errout:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: nrf52_rtc_ackint
|
|
|
|
****************************************************************************/
|
|
|
|
|
2022-04-17 08:01:48 +02:00
|
|
|
static int nrf52_rtc_ackint(struct nrf52_rtc_dev_s *dev, uint8_t s)
|
2020-07-18 10:05:46 +02:00
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
DEBUGASSERT(dev);
|
|
|
|
|
|
|
|
switch (s)
|
|
|
|
{
|
2020-08-24 19:19:41 +02:00
|
|
|
case NRF52_RTC_EVT_TICK:
|
2020-07-18 10:05:46 +02:00
|
|
|
{
|
|
|
|
nrf52_rtc_putreg(dev, NRF52_RTC_EVENTS_TICK_OFFSET, 0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-08-24 19:19:41 +02:00
|
|
|
case NRF52_RTC_EVT_OVRFLW:
|
2020-07-18 10:05:46 +02:00
|
|
|
{
|
|
|
|
nrf52_rtc_putreg(dev, NRF52_RTC_EVENTS_OVRFLW_OFFSET, 0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-08-24 19:19:41 +02:00
|
|
|
case NRF52_RTC_EVT_COMPARE0:
|
2020-07-18 10:05:46 +02:00
|
|
|
{
|
|
|
|
nrf52_rtc_putreg(dev, NRF52_RTC_EVENTS_COMPARE_OFFSET(0), 0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-08-24 19:19:41 +02:00
|
|
|
case NRF52_RTC_EVT_COMPARE1:
|
2020-07-18 10:05:46 +02:00
|
|
|
{
|
|
|
|
nrf52_rtc_putreg(dev, NRF52_RTC_EVENTS_COMPARE_OFFSET(1), 0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-08-24 19:19:41 +02:00
|
|
|
case NRF52_RTC_EVT_COMPARE2:
|
2020-07-18 10:05:46 +02:00
|
|
|
{
|
|
|
|
nrf52_rtc_putreg(dev, NRF52_RTC_EVENTS_COMPARE_OFFSET(2), 0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-08-24 19:19:41 +02:00
|
|
|
case NRF52_RTC_EVT_COMPARE3:
|
2020-07-18 10:05:46 +02:00
|
|
|
{
|
|
|
|
nrf52_rtc_putreg(dev, NRF52_RTC_EVENTS_COMPARE_OFFSET(3), 0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
rtcerr("ERROR: unsupported IRQ source %d\n", s);
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto errout;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
errout:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2020-07-25 03:59:05 +02:00
|
|
|
/****************************************************************************
|
|
|
|
* Name: nrf52_rtc_enableevt
|
|
|
|
****************************************************************************/
|
|
|
|
|
2022-04-17 08:01:48 +02:00
|
|
|
static int nrf52_rtc_enableevt(struct nrf52_rtc_dev_s *dev, uint8_t evt)
|
2020-07-25 03:59:05 +02:00
|
|
|
{
|
|
|
|
uint32_t regval = 0;
|
|
|
|
int ret = OK;
|
|
|
|
|
|
|
|
DEBUGASSERT(dev);
|
|
|
|
|
|
|
|
/* Get register value for given event */
|
|
|
|
|
|
|
|
regval = nrf52_rtc_evt2reg(dev, evt);
|
|
|
|
if (regval == 0)
|
|
|
|
{
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto errout;
|
|
|
|
}
|
|
|
|
|
|
|
|
nrf52_rtc_putreg(dev, NRF52_RTC_EVTENSET_OFFSET, regval);
|
|
|
|
|
|
|
|
errout:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: nrf52_rtc_disableevt
|
|
|
|
****************************************************************************/
|
|
|
|
|
2022-04-17 08:01:48 +02:00
|
|
|
static int nrf52_rtc_disableevt(struct nrf52_rtc_dev_s *dev, uint8_t evt)
|
2020-07-25 03:59:05 +02:00
|
|
|
{
|
|
|
|
uint32_t regval = 0;
|
|
|
|
int ret = OK;
|
|
|
|
|
|
|
|
DEBUGASSERT(dev);
|
|
|
|
|
|
|
|
/* Get register value for given event */
|
|
|
|
|
|
|
|
regval = nrf52_rtc_evt2reg(dev, evt);
|
|
|
|
if (regval == 0)
|
|
|
|
{
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto errout;
|
|
|
|
}
|
|
|
|
|
|
|
|
nrf52_rtc_putreg(dev, NRF52_RTC_EVTENCLR_OFFSET, regval);
|
|
|
|
|
|
|
|
errout:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2021-01-30 21:31:30 +01:00
|
|
|
/****************************************************************************
|
|
|
|
* Name: nrf52_rtc_getbase
|
|
|
|
****************************************************************************/
|
|
|
|
|
2022-04-17 08:01:48 +02:00
|
|
|
static uint32_t nrf52_rtc_getbase(struct nrf52_rtc_dev_s *dev)
|
2021-01-30 21:31:30 +01:00
|
|
|
{
|
2022-04-17 08:01:48 +02:00
|
|
|
struct nrf52_rtc_priv_s *rtc = (struct nrf52_rtc_priv_s *)dev;
|
2021-01-30 21:31:30 +01:00
|
|
|
DEBUGASSERT(dev);
|
|
|
|
|
|
|
|
return rtc->base;
|
|
|
|
}
|
|
|
|
|
2020-07-18 10:05:46 +02:00
|
|
|
/****************************************************************************
|
|
|
|
* Public Functions
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: nrf52_rtc_init
|
|
|
|
*
|
|
|
|
* Description:
|
|
|
|
* Initialize RTC device
|
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
2022-04-17 08:01:48 +02:00
|
|
|
struct nrf52_rtc_dev_s *nrf52_rtc_init(int rtc)
|
2020-07-18 10:05:46 +02:00
|
|
|
{
|
2022-04-17 08:01:48 +02:00
|
|
|
struct nrf52_rtc_priv_s *priv = NULL;
|
2020-07-18 10:05:46 +02:00
|
|
|
|
|
|
|
/* Get RTC instance */
|
|
|
|
|
|
|
|
switch (rtc)
|
|
|
|
{
|
|
|
|
#ifdef CONFIG_NRF52_RTC0
|
|
|
|
case 0:
|
|
|
|
{
|
|
|
|
priv = &g_nrf52_rtc0_priv;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef CONFIG_NRF52_RTC1
|
|
|
|
case 1:
|
|
|
|
{
|
|
|
|
priv = &g_nrf52_rtc1_priv;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef CONFIG_NRF52_RTC2
|
|
|
|
case 2:
|
|
|
|
{
|
|
|
|
priv = &g_nrf52_rtc2_priv;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
rtcerr("ERROR: unsupported RTC %d\n", rtc);
|
|
|
|
goto errout;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (priv->inuse != false)
|
|
|
|
{
|
|
|
|
/* RTC already in use */
|
|
|
|
|
|
|
|
priv = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
errout:
|
2022-04-17 08:01:48 +02:00
|
|
|
return (struct nrf52_rtc_dev_s *)priv;
|
2020-07-18 10:05:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: nrf52_rtc_deinit
|
|
|
|
*
|
|
|
|
* Description:
|
|
|
|
* Deinit RTC device
|
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
2022-04-17 08:01:48 +02:00
|
|
|
int nrf52_rtc_deinit(struct nrf52_rtc_dev_s *dev)
|
2020-07-18 10:05:46 +02:00
|
|
|
{
|
2022-04-17 08:01:48 +02:00
|
|
|
struct nrf52_rtc_priv_s *rtc = NULL;
|
2020-07-18 10:05:46 +02:00
|
|
|
|
|
|
|
DEBUGASSERT(dev);
|
|
|
|
|
2022-04-17 08:01:48 +02:00
|
|
|
rtc = (struct nrf52_rtc_priv_s *)dev;
|
2020-07-18 10:05:46 +02:00
|
|
|
|
|
|
|
rtc->inuse = false;
|
|
|
|
|
|
|
|
return OK;
|
|
|
|
}
|