3b895064af
This API is not widely used and it is actually broken since device runtime power management is not checking it when suspending and resuming. On top of that, this API is very close to pm_device_busy* API, close enough to consolidate in only one API. Signed-off-by: Flavio Ceolin <flavio.ceolin@intel.com>
395 lines
8.7 KiB
C
395 lines
8.7 KiB
C
/*
|
|
* Copyright (c) 2018 Intel Corporation.
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
#include <zephyr/device.h>
|
|
#include <zephyr/pm/device.h>
|
|
#include <zephyr/pm/device_runtime.h>
|
|
#include <zephyr/sys/iterable_sections.h>
|
|
|
|
#include <zephyr/logging/log.h>
|
|
LOG_MODULE_REGISTER(pm_device, CONFIG_PM_DEVICE_LOG_LEVEL);
|
|
|
|
static const enum pm_device_state action_target_state[] = {
|
|
[PM_DEVICE_ACTION_SUSPEND] = PM_DEVICE_STATE_SUSPENDED,
|
|
[PM_DEVICE_ACTION_RESUME] = PM_DEVICE_STATE_ACTIVE,
|
|
[PM_DEVICE_ACTION_TURN_OFF] = PM_DEVICE_STATE_OFF,
|
|
[PM_DEVICE_ACTION_TURN_ON] = PM_DEVICE_STATE_SUSPENDED,
|
|
};
|
|
static const enum pm_device_state action_expected_state[] = {
|
|
[PM_DEVICE_ACTION_SUSPEND] = PM_DEVICE_STATE_ACTIVE,
|
|
[PM_DEVICE_ACTION_RESUME] = PM_DEVICE_STATE_SUSPENDED,
|
|
[PM_DEVICE_ACTION_TURN_OFF] = PM_DEVICE_STATE_SUSPENDED,
|
|
[PM_DEVICE_ACTION_TURN_ON] = PM_DEVICE_STATE_OFF,
|
|
};
|
|
|
|
const char *pm_device_state_str(enum pm_device_state state)
|
|
{
|
|
switch (state) {
|
|
case PM_DEVICE_STATE_ACTIVE:
|
|
return "active";
|
|
case PM_DEVICE_STATE_SUSPENDED:
|
|
return "suspended";
|
|
case PM_DEVICE_STATE_OFF:
|
|
return "off";
|
|
default:
|
|
return "";
|
|
}
|
|
}
|
|
|
|
int pm_device_action_run(const struct device *dev,
|
|
enum pm_device_action action)
|
|
{
|
|
struct pm_device_base *pm = dev->pm_base;
|
|
int ret;
|
|
|
|
if (pm == NULL) {
|
|
return -ENOSYS;
|
|
}
|
|
|
|
/* Validate action against current state */
|
|
if (pm->state == action_target_state[action]) {
|
|
return -EALREADY;
|
|
}
|
|
if (pm->state != action_expected_state[action]) {
|
|
return -ENOTSUP;
|
|
}
|
|
|
|
ret = pm->action_cb(dev, action);
|
|
if (ret < 0) {
|
|
/*
|
|
* TURN_ON and TURN_OFF are actions triggered by a power domain
|
|
* when it is resumed or suspended, which means that the energy
|
|
* to the device will be removed or added. For this reason, if
|
|
* the transition fails or the device does not handle these
|
|
* actions its state still needs to updated to reflect its
|
|
* physical behavior.
|
|
*
|
|
* The function will still return the error code so the domain
|
|
* can take whatever action is more appropriated.
|
|
*/
|
|
switch (action) {
|
|
case PM_DEVICE_ACTION_TURN_ON:
|
|
/* Store an error flag when the transition explicitly fails */
|
|
if (ret != -ENOTSUP) {
|
|
atomic_set_bit(&pm->flags, PM_DEVICE_FLAG_TURN_ON_FAILED);
|
|
}
|
|
__fallthrough;
|
|
case PM_DEVICE_ACTION_TURN_OFF:
|
|
pm->state = action_target_state[action];
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
pm->state = action_target_state[action];
|
|
/* Power up flags are no longer relevant */
|
|
if (action == PM_DEVICE_ACTION_TURN_OFF) {
|
|
atomic_clear_bit(&pm->flags, PM_DEVICE_FLAG_PD_CLAIMED);
|
|
atomic_clear_bit(&pm->flags, PM_DEVICE_FLAG_TURN_ON_FAILED);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int power_domain_add_or_remove(const struct device *dev,
|
|
const struct device *domain,
|
|
bool add)
|
|
{
|
|
#if defined(CONFIG_DEVICE_DEPS_DYNAMIC)
|
|
device_handle_t *rv = domain->deps;
|
|
device_handle_t dev_handle = -1;
|
|
size_t i = 0, region = 0;
|
|
|
|
/*
|
|
* Supported devices are stored as device handle and not
|
|
* device pointers. So, it is necessary to find what is
|
|
* the handle associated to the given device.
|
|
*/
|
|
STRUCT_SECTION_FOREACH(device, iter_dev) {
|
|
if (iter_dev == dev) {
|
|
dev_handle = i + 1;
|
|
break;
|
|
}
|
|
|
|
i++;
|
|
}
|
|
|
|
/*
|
|
* The last part is to find an available slot in the
|
|
* supported section of handles array and replace it
|
|
* with the device handle.
|
|
*/
|
|
while (region != 2) {
|
|
if (*rv == Z_DEVICE_DEPS_SEP) {
|
|
region++;
|
|
}
|
|
rv++;
|
|
}
|
|
|
|
i = 0;
|
|
while (rv[i] != Z_DEVICE_DEPS_ENDS) {
|
|
if (add == false) {
|
|
if (rv[i] == dev_handle) {
|
|
dev->pm_base->domain = NULL;
|
|
rv[i] = DEVICE_HANDLE_NULL;
|
|
return 0;
|
|
}
|
|
} else {
|
|
if (rv[i] == DEVICE_HANDLE_NULL) {
|
|
dev->pm_base->domain = domain;
|
|
rv[i] = dev_handle;
|
|
return 0;
|
|
}
|
|
}
|
|
++i;
|
|
}
|
|
|
|
return add ? -ENOSPC : -ENOENT;
|
|
#else
|
|
ARG_UNUSED(dev);
|
|
ARG_UNUSED(domain);
|
|
ARG_UNUSED(add);
|
|
|
|
return -ENOSYS;
|
|
#endif
|
|
}
|
|
|
|
int pm_device_power_domain_remove(const struct device *dev,
|
|
const struct device *domain)
|
|
{
|
|
return power_domain_add_or_remove(dev, domain, false);
|
|
}
|
|
|
|
int pm_device_power_domain_add(const struct device *dev,
|
|
const struct device *domain)
|
|
{
|
|
return power_domain_add_or_remove(dev, domain, true);
|
|
}
|
|
|
|
#ifdef CONFIG_DEVICE_DEPS
|
|
struct pm_visitor_context {
|
|
pm_device_action_failed_cb_t failure_cb;
|
|
enum pm_device_action action;
|
|
};
|
|
|
|
static int pm_device_children_visitor(const struct device *dev, void *context)
|
|
{
|
|
struct pm_visitor_context *visitor_context = context;
|
|
int rc;
|
|
|
|
rc = pm_device_action_run(dev, visitor_context->action);
|
|
if ((visitor_context->failure_cb != NULL) && (rc < 0)) {
|
|
/* Stop the iteration if the callback requests it */
|
|
if (!visitor_context->failure_cb(dev, rc)) {
|
|
return rc;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void pm_device_children_action_run(const struct device *dev,
|
|
enum pm_device_action action,
|
|
pm_device_action_failed_cb_t failure_cb)
|
|
{
|
|
struct pm_visitor_context visitor_context = {
|
|
.failure_cb = failure_cb,
|
|
.action = action
|
|
};
|
|
|
|
(void)device_supported_foreach(dev, pm_device_children_visitor, &visitor_context);
|
|
}
|
|
#endif
|
|
|
|
int pm_device_state_get(const struct device *dev,
|
|
enum pm_device_state *state)
|
|
{
|
|
struct pm_device_base *pm = dev->pm_base;
|
|
|
|
if (pm == NULL) {
|
|
return -ENOSYS;
|
|
}
|
|
|
|
*state = pm->state;
|
|
|
|
return 0;
|
|
}
|
|
|
|
bool pm_device_is_any_busy(void)
|
|
{
|
|
const struct device *devs;
|
|
size_t devc;
|
|
|
|
devc = z_device_get_all_static(&devs);
|
|
|
|
for (const struct device *dev = devs; dev < (devs + devc); dev++) {
|
|
struct pm_device_base *pm = dev->pm_base;
|
|
|
|
if (pm == NULL) {
|
|
continue;
|
|
}
|
|
|
|
if (atomic_test_bit(&pm->flags, PM_DEVICE_FLAG_BUSY)) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool pm_device_is_busy(const struct device *dev)
|
|
{
|
|
struct pm_device_base *pm = dev->pm_base;
|
|
|
|
if (pm == NULL) {
|
|
return false;
|
|
}
|
|
|
|
return atomic_test_bit(&pm->flags, PM_DEVICE_FLAG_BUSY);
|
|
}
|
|
|
|
void pm_device_busy_set(const struct device *dev)
|
|
{
|
|
struct pm_device_base *pm = dev->pm_base;
|
|
|
|
if (pm == NULL) {
|
|
return;
|
|
}
|
|
|
|
atomic_set_bit(&pm->flags, PM_DEVICE_FLAG_BUSY);
|
|
}
|
|
|
|
void pm_device_busy_clear(const struct device *dev)
|
|
{
|
|
struct pm_device_base *pm = dev->pm_base;
|
|
|
|
if (pm == NULL) {
|
|
return;
|
|
}
|
|
|
|
atomic_clear_bit(&pm->flags, PM_DEVICE_FLAG_BUSY);
|
|
}
|
|
|
|
bool pm_device_wakeup_enable(const struct device *dev, bool enable)
|
|
{
|
|
atomic_val_t flags, new_flags;
|
|
struct pm_device_base *pm = dev->pm_base;
|
|
|
|
if (pm == NULL) {
|
|
return false;
|
|
}
|
|
|
|
flags = atomic_get(&pm->flags);
|
|
|
|
if ((flags & BIT(PM_DEVICE_FLAG_WS_CAPABLE)) == 0U) {
|
|
return false;
|
|
}
|
|
|
|
if (enable) {
|
|
new_flags = flags |
|
|
BIT(PM_DEVICE_FLAG_WS_ENABLED);
|
|
} else {
|
|
new_flags = flags & ~BIT(PM_DEVICE_FLAG_WS_ENABLED);
|
|
}
|
|
|
|
return atomic_cas(&pm->flags, flags, new_flags);
|
|
}
|
|
|
|
bool pm_device_wakeup_is_enabled(const struct device *dev)
|
|
{
|
|
struct pm_device_base *pm = dev->pm_base;
|
|
|
|
if (pm == NULL) {
|
|
return false;
|
|
}
|
|
|
|
return atomic_test_bit(&pm->flags,
|
|
PM_DEVICE_FLAG_WS_ENABLED);
|
|
}
|
|
|
|
bool pm_device_wakeup_is_capable(const struct device *dev)
|
|
{
|
|
struct pm_device_base *pm = dev->pm_base;
|
|
|
|
if (pm == NULL) {
|
|
return false;
|
|
}
|
|
|
|
return atomic_test_bit(&pm->flags,
|
|
PM_DEVICE_FLAG_WS_CAPABLE);
|
|
}
|
|
|
|
bool pm_device_on_power_domain(const struct device *dev)
|
|
{
|
|
#ifdef CONFIG_PM_DEVICE_POWER_DOMAIN
|
|
struct pm_device_base *pm = dev->pm_base;
|
|
|
|
if (pm == NULL) {
|
|
return false;
|
|
}
|
|
return pm->domain != NULL;
|
|
#else
|
|
ARG_UNUSED(dev);
|
|
return false;
|
|
#endif
|
|
}
|
|
|
|
bool pm_device_is_powered(const struct device *dev)
|
|
{
|
|
#ifdef CONFIG_PM_DEVICE_POWER_DOMAIN
|
|
struct pm_device_base *pm = dev->pm_base;
|
|
|
|
/* If a device doesn't support PM or is not under a PM domain,
|
|
* assume it is always powered on.
|
|
*/
|
|
return (pm == NULL) ||
|
|
(pm->domain == NULL) ||
|
|
(pm->domain->pm_base->state == PM_DEVICE_STATE_ACTIVE);
|
|
#else
|
|
ARG_UNUSED(dev);
|
|
return true;
|
|
#endif
|
|
}
|
|
|
|
int pm_device_driver_init(const struct device *dev,
|
|
pm_device_action_cb_t action_cb)
|
|
{
|
|
struct pm_device_base *pm = dev->pm_base;
|
|
int rc = 0;
|
|
|
|
/* Work only needs to be performed if the device is powered */
|
|
if (pm_device_is_powered(dev)) {
|
|
/* Run power-up logic */
|
|
rc = action_cb(dev, PM_DEVICE_ACTION_TURN_ON);
|
|
if (rc != 0) {
|
|
return rc;
|
|
}
|
|
/* If device has no PM structure */
|
|
if (pm == NULL) {
|
|
/* Device should always be active */
|
|
return action_cb(dev, PM_DEVICE_ACTION_RESUME);
|
|
}
|
|
/* If device will have PM device runtime enabled */
|
|
if (IS_ENABLED(CONFIG_PM_DEVICE_RUNTIME) &&
|
|
atomic_test_bit(&pm->flags, PM_DEVICE_FLAG_RUNTIME_AUTO)) {
|
|
/* Init into suspend mode.
|
|
* This saves a SUSPENDED->ACTIVE->SUSPENDED cycle.
|
|
*/
|
|
pm_device_init_suspended(dev);
|
|
}
|
|
/* No PM enabled on the device by default */
|
|
else {
|
|
/* Startup into active mode */
|
|
return action_cb(dev, PM_DEVICE_ACTION_RESUME);
|
|
}
|
|
} else {
|
|
/* Start in off mode */
|
|
pm_device_init_off(dev);
|
|
}
|
|
return rc;
|
|
}
|