mirror of
https://github.com/apache/nuttx.git
synced 2026-06-06 08:36:24 +08:00
nrf52: add SPI PM support (disable/enable SPI peripheral on sleep)
This commit is contained in:
committed by
Mateusz Szafoni
parent
a806ca9577
commit
13619ea0df
@@ -159,3 +159,7 @@ endif
|
||||
ifeq ($(CONFIG_NRF52_SAADC),y)
|
||||
CHIP_CSRCS += nrf52_adc.c
|
||||
endif
|
||||
|
||||
ifeq ($(CONFIG_PM),y)
|
||||
CHIP_CSRCS += nrf52_pminitialize.c
|
||||
endif
|
||||
|
||||
@@ -0,0 +1,55 @@
|
||||
/****************************************************************************
|
||||
* arch/arm/src/nrf52/nrf52_pminitialize.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>
|
||||
#include <nuttx/power/pm.h>
|
||||
|
||||
/****************************************************************************
|
||||
* Public Functions
|
||||
****************************************************************************/
|
||||
|
||||
/****************************************************************************
|
||||
* Name: arm_pminitialize
|
||||
*
|
||||
* Description:
|
||||
* This function is called by MCU-specific logic at power-on reset in
|
||||
* order to provide one-time initialization the power management subsystem.
|
||||
* This function must be called *very* early in the initialization sequence
|
||||
* *before* any other device drivers are initialized (since they may
|
||||
* attempt to register with the power management subsystem).
|
||||
*
|
||||
* Input Parameters:
|
||||
* None.
|
||||
*
|
||||
* Returned Value:
|
||||
* None.
|
||||
*
|
||||
****************************************************************************/
|
||||
|
||||
void arm_pminitialize(void)
|
||||
{
|
||||
/* Then initialize the NuttX power management subsystem proper */
|
||||
|
||||
pm_initialize();
|
||||
}
|
||||
@@ -31,6 +31,7 @@
|
||||
#include <nuttx/arch.h>
|
||||
#include <nuttx/semaphore.h>
|
||||
#include <arch/board/board.h>
|
||||
#include <nuttx/power/pm.h>
|
||||
|
||||
#include "arm_arch.h"
|
||||
#include "barriers.h"
|
||||
@@ -86,6 +87,7 @@ struct nrf52_spidev_s
|
||||
#ifdef CONFIG_NRF52_SPI_MASTER_INTERRUPTS
|
||||
sem_t sem_isr; /* Interrupt wait semaphore */
|
||||
#endif
|
||||
bool initialized;
|
||||
};
|
||||
|
||||
/****************************************************************************
|
||||
@@ -133,12 +135,29 @@ static int nrf52_spi_init(FAR struct nrf52_spidev_s *priv);
|
||||
static void nrf52_spi_pselinit(FAR struct nrf52_spidev_s *priv,
|
||||
uint32_t offset, nrf52_pinset_t pinset);
|
||||
static void nrf52_spi_gpioinit(FAR struct nrf52_spidev_s *priv);
|
||||
|
||||
#ifdef CONFIG_PM
|
||||
static int nrf52_spi_deinit(FAR struct nrf52_spidev_s *priv);
|
||||
static void nrf52_spi_gpiodeinit(FAR struct nrf52_spidev_s *priv);
|
||||
|
||||
static int nrf52_spi_pm_prepare(FAR struct pm_callback_s *cb, int domain,
|
||||
enum pm_state_e pmstate);
|
||||
static void nrf52_spi_pm_notify(FAR struct pm_callback_s *cb, int domain,
|
||||
enum pm_state_e pmstate);
|
||||
#endif
|
||||
|
||||
/****************************************************************************
|
||||
* Private Data
|
||||
****************************************************************************/
|
||||
|
||||
#ifdef CONFIG_PM
|
||||
struct pm_callback_s g_pm_callbacks =
|
||||
{
|
||||
.prepare = nrf52_spi_pm_prepare,
|
||||
.notify = nrf52_spi_pm_notify
|
||||
};
|
||||
#endif
|
||||
|
||||
/* SPI0 */
|
||||
|
||||
#ifdef CONFIG_NRF52_SPI0_MASTER
|
||||
@@ -439,7 +458,7 @@ static int nrf52_spi_init(FAR struct nrf52_spidev_s *priv)
|
||||
return OK;
|
||||
}
|
||||
|
||||
|
||||
#ifdef CONFIG_PM
|
||||
/****************************************************************************
|
||||
* Name: nrf52_spi_deinit
|
||||
*
|
||||
@@ -471,6 +490,7 @@ static int nrf52_spi_deinit(FAR struct nrf52_spidev_s *priv)
|
||||
|
||||
return OK;
|
||||
}
|
||||
#endif
|
||||
|
||||
/****************************************************************************
|
||||
* Name: nrf52_spi_pselinit
|
||||
@@ -574,6 +594,7 @@ static void nrf52_spi_gpioinit(FAR struct nrf52_spidev_s *priv)
|
||||
#endif
|
||||
}
|
||||
|
||||
#ifdef CONFIG_PM
|
||||
/****************************************************************************
|
||||
* Name: nrf52_spi_gpioinit
|
||||
*
|
||||
@@ -634,6 +655,7 @@ static void nrf52_spi_gpiodeinit(FAR struct nrf52_spidev_s *priv)
|
||||
}
|
||||
#endif
|
||||
}
|
||||
#endif
|
||||
|
||||
/****************************************************************************
|
||||
* Name: nrf52_spi_lock
|
||||
@@ -1079,7 +1101,7 @@ static void nrf52_spi_exchange(FAR struct spi_dev_s *dev,
|
||||
* in batches
|
||||
*/
|
||||
|
||||
if (nwords > 0xFF)
|
||||
if (nwords > 0xff)
|
||||
{
|
||||
if (rxbuffer != NULL)
|
||||
{
|
||||
@@ -1159,7 +1181,7 @@ static void nrf52_spi_exchange(FAR struct spi_dev_s *dev,
|
||||
|
||||
/* Clear list mode */
|
||||
|
||||
if (nwords > 0xFF)
|
||||
if (nwords > 0xff)
|
||||
{
|
||||
nrf52_spi_putreg(priv, NRF52_SPIM_RXDLIST_OFFSET, 0);
|
||||
nrf52_spi_putreg(priv, NRF52_SPIM_TXDLIST_OFFSET, 0);
|
||||
@@ -1251,6 +1273,126 @@ static int nrf52_spi_trigger(FAR struct spi_dev_s *dev)
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef CONFIG_PM
|
||||
/****************************************************************************
|
||||
* Name: nrf52_spi_pm_prepare
|
||||
****************************************************************************/
|
||||
|
||||
static int nrf52_spi_pm_prepare(FAR struct pm_callback_s *cb, int domain,
|
||||
enum pm_state_e pmstate)
|
||||
{
|
||||
if (pmstate == PM_STANDBY || pmstate == PM_SLEEP)
|
||||
{
|
||||
bool active = false;
|
||||
|
||||
#ifdef CONFIG_NRF52_SPI0_MASTER
|
||||
active |= nrf52_spi_getreg(&g_spi0dev, SPIM_EVENTS_STARTED);
|
||||
#endif
|
||||
#ifdef CONFIG_NRF52_SPI1_MASTER
|
||||
active |= nrf52_spi_getreg(&g_spi0dev, SPIM_EVENTS_STARTED);
|
||||
#endif
|
||||
#ifdef CONFIG_NRF52_SPI2_MASTER
|
||||
active |= nrf52_spi_getreg(&g_spi0dev, SPIM_EVENTS_STARTED);
|
||||
#endif
|
||||
#ifdef CONFIG_NRF52_SPI3_MASTER
|
||||
active |= nrf52_spi_getreg(&g_spi0dev, SPIM_EVENTS_STARTED);
|
||||
#endif
|
||||
|
||||
if (active)
|
||||
{
|
||||
/* SPI is being used, cannot disable */
|
||||
|
||||
return -1;
|
||||
}
|
||||
else
|
||||
{
|
||||
/* SPI is inactive, can go to sleep */
|
||||
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
/* We can always go to any other state */
|
||||
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
|
||||
/****************************************************************************
|
||||
* Name: nrf52_spi_pm_notify
|
||||
****************************************************************************/
|
||||
|
||||
static void nrf52_spi_pm_notify(FAR struct pm_callback_s *cb, int domain,
|
||||
enum pm_state_e pmstate)
|
||||
{
|
||||
if (pmstate == PM_SLEEP || pmstate == PM_STANDBY)
|
||||
{
|
||||
/* Deinit SPI peripheral on each initialized device */
|
||||
|
||||
#ifdef CONFIG_NRF52_SPI0_MASTER
|
||||
if (g_spi0dev.initialized)
|
||||
{
|
||||
nrf52_spi_deinit(&g_spi0dev);
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef CONFIG_NRF52_SPI1_MASTER
|
||||
if (g_spi1dev.initialized)
|
||||
{
|
||||
nrf52_spi_deinit(&g_spi1dev);
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef CONFIG_NRF52_SPI2_MASTER
|
||||
if (g_spi2dev.initialized)
|
||||
{
|
||||
nrf52_spi_deinit(&g_spi2dev);
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef CONFIG_NRF52_SPI3_MASTER
|
||||
if (g_spi3dev.initialized)
|
||||
{
|
||||
nrf52_spi_deinit(&g_spi3dev);
|
||||
}
|
||||
#endif
|
||||
}
|
||||
else
|
||||
{
|
||||
/* Reinit SPI peripheral on each initialized device */
|
||||
|
||||
#ifdef CONFIG_NRF52_SPI0_MASTER
|
||||
if (g_spi0dev.initialized)
|
||||
{
|
||||
nrf52_spi_init(&g_spi0dev);
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef CONFIG_NRF52_SPI1_MASTER
|
||||
if (g_spi1dev.initialized)
|
||||
{
|
||||
nrf52_spi_init(&g_spi1dev);
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef CONFIG_NRF52_SPI2_MASTER
|
||||
if (g_spi2dev.initialized)
|
||||
{
|
||||
nrf52_spi_init(&g_spi2dev);
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef CONFIG_NRF52_SPI3_MASTER
|
||||
if (g_spi3dev.initialized)
|
||||
{
|
||||
nrf52_spi_init(&g_spi3dev);
|
||||
}
|
||||
#endif
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
/****************************************************************************
|
||||
* Public Functions
|
||||
****************************************************************************/
|
||||
@@ -1319,6 +1461,10 @@ FAR struct spi_dev_s *nrf52_spibus_initialize(int port)
|
||||
|
||||
nrf52_spi_init(priv);
|
||||
|
||||
/* Mark device as initialized */
|
||||
|
||||
priv->initialized = true;
|
||||
|
||||
/* Initialize the SPI semaphore */
|
||||
|
||||
nxsem_init(&priv->exclsem, 0, 1);
|
||||
|
||||
@@ -232,7 +232,7 @@ struct pm_governor_s
|
||||
*
|
||||
* NOTE: since this will be called from pm_initialize(), the system
|
||||
* is in very early boot state when this callback is invoked. Thus,
|
||||
* only ver basic initialization should be performed (e.g. no memory
|
||||
* only very basic initialization should be performed (e.g. no memory
|
||||
* should be allocated).
|
||||
*
|
||||
**************************************************************************/
|
||||
|
||||
Reference in New Issue
Block a user