2018-08-16 07:09:40 +02:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2018 Intel Corporation.
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <device.h>
|
2021-10-13 15:39:40 +02:00
|
|
|
#include <pm/device.h>
|
2021-11-18 06:50:08 +01:00
|
|
|
#include <pm/device_runtime.h>
|
2018-08-16 07:09:40 +02:00
|
|
|
|
|
|
|
#include <logging/log.h>
|
2021-10-27 13:39:12 +02:00
|
|
|
LOG_MODULE_REGISTER(pm_device, CONFIG_PM_DEVICE_LOG_LEVEL);
|
2018-08-16 07:09:40 +02:00
|
|
|
|
2021-06-03 19:06:53 +02:00
|
|
|
const char *pm_device_state_str(enum pm_device_state state)
|
2021-01-25 15:08:48 +01:00
|
|
|
{
|
|
|
|
switch (state) {
|
2021-05-07 23:18:57 +02:00
|
|
|
case PM_DEVICE_STATE_ACTIVE:
|
2021-01-25 15:08:48 +01:00
|
|
|
return "active";
|
2021-07-02 18:09:07 +02:00
|
|
|
case PM_DEVICE_STATE_SUSPENDED:
|
|
|
|
return "suspended";
|
2021-05-07 23:18:57 +02:00
|
|
|
case PM_DEVICE_STATE_OFF:
|
2021-01-25 15:08:48 +01:00
|
|
|
return "off";
|
|
|
|
default:
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
}
|
2021-05-03 18:12:17 +02:00
|
|
|
|
2021-11-23 00:32:17 +01:00
|
|
|
int pm_device_action_run(const struct device *dev,
|
2022-01-06 12:17:38 +01:00
|
|
|
enum pm_device_action action)
|
2021-11-23 00:32:17 +01:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
enum pm_device_state state;
|
|
|
|
struct pm_device *pm = dev->pm;
|
|
|
|
|
|
|
|
if (pm == NULL) {
|
|
|
|
return -ENOSYS;
|
|
|
|
}
|
|
|
|
|
2021-12-15 21:06:34 +01:00
|
|
|
if (pm_device_state_is_locked(dev)) {
|
|
|
|
return -EPERM;
|
|
|
|
}
|
|
|
|
|
2021-11-23 00:32:17 +01:00
|
|
|
switch (action) {
|
|
|
|
case PM_DEVICE_ACTION_FORCE_SUSPEND:
|
|
|
|
__fallthrough;
|
|
|
|
case PM_DEVICE_ACTION_SUSPEND:
|
|
|
|
if (pm->state == PM_DEVICE_STATE_SUSPENDED) {
|
|
|
|
return -EALREADY;
|
|
|
|
} else if (pm->state == PM_DEVICE_STATE_OFF) {
|
|
|
|
return -ENOTSUP;
|
|
|
|
}
|
|
|
|
|
|
|
|
state = PM_DEVICE_STATE_SUSPENDED;
|
|
|
|
break;
|
|
|
|
case PM_DEVICE_ACTION_RESUME:
|
|
|
|
if (pm->state == PM_DEVICE_STATE_ACTIVE) {
|
|
|
|
return -EALREADY;
|
|
|
|
}
|
|
|
|
|
|
|
|
state = PM_DEVICE_STATE_ACTIVE;
|
|
|
|
break;
|
|
|
|
case PM_DEVICE_ACTION_TURN_OFF:
|
|
|
|
if (pm->state == PM_DEVICE_STATE_OFF) {
|
|
|
|
return -EALREADY;
|
|
|
|
}
|
|
|
|
|
|
|
|
state = PM_DEVICE_STATE_OFF;
|
|
|
|
break;
|
2022-01-05 08:18:37 +01:00
|
|
|
case PM_DEVICE_ACTION_TURN_ON:
|
|
|
|
if (pm->state != PM_DEVICE_STATE_OFF) {
|
|
|
|
return -ENOTSUP;
|
|
|
|
}
|
|
|
|
|
|
|
|
state = PM_DEVICE_STATE_SUSPENDED;
|
|
|
|
break;
|
2021-11-23 00:32:17 +01:00
|
|
|
default:
|
|
|
|
return -ENOTSUP;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = pm->action_cb(dev, action);
|
|
|
|
if (ret < 0) {
|
2022-01-14 00:12:37 +01:00
|
|
|
/*
|
|
|
|
* 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, even
|
|
|
|
* if the device does not handle these actions its state 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.
|
|
|
|
*/
|
|
|
|
if ((ret == -ENOTSUP) && ((action == PM_DEVICE_ACTION_TURN_ON)
|
|
|
|
|| (action == PM_DEVICE_ACTION_TURN_OFF))) {
|
|
|
|
pm->state = state;
|
|
|
|
}
|
2021-06-04 17:41:39 +02:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2021-10-13 15:48:22 +02:00
|
|
|
pm->state = state;
|
2021-06-04 17:41:39 +02:00
|
|
|
|
|
|
|
return 0;
|
2021-05-03 18:12:17 +02:00
|
|
|
}
|
|
|
|
|
2022-01-05 08:45:10 +01:00
|
|
|
void pm_device_children_action_run(const struct device *dev,
|
2022-01-06 12:17:38 +01:00
|
|
|
enum pm_device_action action,
|
|
|
|
pm_device_action_failed_cb_t failure_cb)
|
2022-01-05 08:45:10 +01:00
|
|
|
{
|
|
|
|
const device_handle_t *handles;
|
|
|
|
size_t handle_count = 0U;
|
|
|
|
int rc = 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We don't use device_supported_foreach here because we don't want the
|
|
|
|
* early exit behaviour of that function. Even if the N'th device fails
|
|
|
|
* to PM_DEVICE_ACTION_TURN_ON for example, we still want to run the
|
|
|
|
* action on the N+1'th device.
|
|
|
|
*/
|
|
|
|
handles = device_supported_handles_get(dev, &handle_count);
|
|
|
|
|
|
|
|
for (size_t i = 0U; i < handle_count; ++i) {
|
|
|
|
device_handle_t dh = handles[i];
|
|
|
|
const struct device *cdev = device_from_handle(dh);
|
|
|
|
|
|
|
|
rc = pm_device_action_run(cdev, action);
|
|
|
|
if ((failure_cb != NULL) && (rc < 0)) {
|
|
|
|
/* Stop the iteration if the callback requests it */
|
|
|
|
if (!failure_cb(cdev, rc)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-03 19:06:53 +02:00
|
|
|
int pm_device_state_get(const struct device *dev,
|
2021-06-04 17:41:39 +02:00
|
|
|
enum pm_device_state *state)
|
2021-05-03 18:12:17 +02:00
|
|
|
{
|
2021-10-13 15:48:22 +02:00
|
|
|
struct pm_device *pm = dev->pm;
|
|
|
|
|
2021-10-13 12:11:40 +02:00
|
|
|
if (pm == NULL) {
|
2021-05-03 18:12:17 +02:00
|
|
|
return -ENOSYS;
|
|
|
|
}
|
|
|
|
|
2021-10-13 15:48:22 +02:00
|
|
|
*state = pm->state;
|
2021-06-04 17:41:39 +02:00
|
|
|
|
|
|
|
return 0;
|
2021-05-03 18:12:17 +02:00
|
|
|
}
|
2021-05-31 15:24:34 +02:00
|
|
|
|
2021-07-29 11:07:27 +02:00
|
|
|
bool pm_device_is_any_busy(void)
|
2021-05-31 15:24:34 +02:00
|
|
|
{
|
2021-08-04 16:59:26 +02:00
|
|
|
const struct device *devs;
|
|
|
|
size_t devc;
|
|
|
|
|
|
|
|
devc = z_device_get_all_static(&devs);
|
2021-05-31 15:24:34 +02:00
|
|
|
|
2021-08-04 16:59:26 +02:00
|
|
|
for (const struct device *dev = devs; dev < (devs + devc); dev++) {
|
2021-10-13 15:48:22 +02:00
|
|
|
struct pm_device *pm = dev->pm;
|
|
|
|
|
2021-10-13 12:11:40 +02:00
|
|
|
if (pm == NULL) {
|
2021-10-11 22:33:38 +02:00
|
|
|
continue;
|
|
|
|
}
|
2021-10-13 15:48:22 +02:00
|
|
|
|
|
|
|
if (atomic_test_bit(&pm->flags, PM_DEVICE_FLAG_BUSY)) {
|
2021-07-29 11:07:27 +02:00
|
|
|
return true;
|
2021-05-31 15:24:34 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-29 11:07:27 +02:00
|
|
|
return false;
|
2021-05-31 15:24:34 +02:00
|
|
|
}
|
|
|
|
|
2021-07-29 11:07:27 +02:00
|
|
|
bool pm_device_is_busy(const struct device *dev)
|
2021-05-31 15:24:34 +02:00
|
|
|
{
|
2021-10-13 15:48:22 +02:00
|
|
|
struct pm_device *pm = dev->pm;
|
|
|
|
|
2021-10-13 12:11:40 +02:00
|
|
|
if (pm == NULL) {
|
2021-10-11 22:33:38 +02:00
|
|
|
return false;
|
|
|
|
}
|
2021-10-13 15:48:22 +02:00
|
|
|
|
|
|
|
return atomic_test_bit(&pm->flags, PM_DEVICE_FLAG_BUSY);
|
2021-05-31 15:24:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void pm_device_busy_set(const struct device *dev)
|
|
|
|
{
|
2021-10-13 15:48:22 +02:00
|
|
|
struct pm_device *pm = dev->pm;
|
|
|
|
|
2021-10-13 12:11:40 +02:00
|
|
|
if (pm == NULL) {
|
2021-10-11 22:33:38 +02:00
|
|
|
return;
|
|
|
|
}
|
2021-10-13 15:48:22 +02:00
|
|
|
|
|
|
|
atomic_set_bit(&pm->flags, PM_DEVICE_FLAG_BUSY);
|
2021-05-31 15:24:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void pm_device_busy_clear(const struct device *dev)
|
|
|
|
{
|
2021-10-13 15:48:22 +02:00
|
|
|
struct pm_device *pm = dev->pm;
|
|
|
|
|
2021-10-13 12:11:40 +02:00
|
|
|
if (pm == NULL) {
|
2021-10-11 22:33:38 +02:00
|
|
|
return;
|
|
|
|
}
|
2021-10-13 15:48:22 +02:00
|
|
|
|
|
|
|
atomic_clear_bit(&pm->flags, PM_DEVICE_FLAG_BUSY);
|
2021-05-31 15:24:34 +02:00
|
|
|
}
|
2021-06-30 23:32:56 +02:00
|
|
|
|
2022-01-06 14:37:24 +01:00
|
|
|
bool pm_device_wakeup_enable(const struct device *dev, bool enable)
|
2021-06-30 23:32:56 +02:00
|
|
|
{
|
|
|
|
atomic_val_t flags, new_flags;
|
2021-10-13 15:48:22 +02:00
|
|
|
struct pm_device *pm = dev->pm;
|
2021-06-30 23:32:56 +02:00
|
|
|
|
2021-10-13 12:11:40 +02:00
|
|
|
if (pm == NULL) {
|
2021-10-11 22:33:38 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-01-06 12:17:38 +01:00
|
|
|
flags = atomic_get(&pm->flags);
|
2021-06-30 23:32:56 +02:00
|
|
|
|
2021-10-29 11:42:16 +02:00
|
|
|
if ((flags & BIT(PM_DEVICE_FLAG_WS_CAPABLE)) == 0U) {
|
2021-06-30 23:32:56 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (enable) {
|
|
|
|
new_flags = flags |
|
2022-01-06 12:17:38 +01:00
|
|
|
BIT(PM_DEVICE_FLAG_WS_ENABLED);
|
2021-06-30 23:32:56 +02:00
|
|
|
} else {
|
2021-10-29 11:42:16 +02:00
|
|
|
new_flags = flags & ~BIT(PM_DEVICE_FLAG_WS_ENABLED);
|
2021-06-30 23:32:56 +02:00
|
|
|
}
|
|
|
|
|
2021-10-13 15:48:22 +02:00
|
|
|
return atomic_cas(&pm->flags, flags, new_flags);
|
2021-06-30 23:32:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
bool pm_device_wakeup_is_enabled(const struct device *dev)
|
|
|
|
{
|
2021-10-13 15:48:22 +02:00
|
|
|
struct pm_device *pm = dev->pm;
|
|
|
|
|
2021-10-13 12:11:40 +02:00
|
|
|
if (pm == NULL) {
|
2021-10-11 22:33:38 +02:00
|
|
|
return false;
|
|
|
|
}
|
2021-10-13 15:48:22 +02:00
|
|
|
|
|
|
|
return atomic_test_bit(&pm->flags,
|
2021-10-29 11:42:16 +02:00
|
|
|
PM_DEVICE_FLAG_WS_ENABLED);
|
2021-06-30 23:32:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
bool pm_device_wakeup_is_capable(const struct device *dev)
|
|
|
|
{
|
2021-10-13 15:48:22 +02:00
|
|
|
struct pm_device *pm = dev->pm;
|
|
|
|
|
2021-10-13 12:11:40 +02:00
|
|
|
if (pm == NULL) {
|
2021-10-11 22:33:38 +02:00
|
|
|
return false;
|
|
|
|
}
|
2021-10-13 15:48:22 +02:00
|
|
|
|
|
|
|
return atomic_test_bit(&pm->flags,
|
2021-10-29 11:42:16 +02:00
|
|
|
PM_DEVICE_FLAG_WS_CAPABLE);
|
2021-06-30 23:32:56 +02:00
|
|
|
}
|
2021-11-18 01:06:10 +01:00
|
|
|
|
|
|
|
void pm_device_state_lock(const struct device *dev)
|
|
|
|
{
|
|
|
|
struct pm_device *pm = dev->pm;
|
|
|
|
|
2021-11-18 06:50:08 +01:00
|
|
|
if ((pm != NULL) && !pm_device_runtime_is_enabled(dev)) {
|
2021-11-18 01:06:10 +01:00
|
|
|
atomic_set_bit(&pm->flags, PM_DEVICE_FLAG_STATE_LOCKED);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void pm_device_state_unlock(const struct device *dev)
|
|
|
|
{
|
|
|
|
struct pm_device *pm = dev->pm;
|
|
|
|
|
|
|
|
if (pm != NULL) {
|
|
|
|
atomic_clear_bit(&pm->flags, PM_DEVICE_FLAG_STATE_LOCKED);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool pm_device_state_is_locked(const struct device *dev)
|
|
|
|
{
|
|
|
|
struct pm_device *pm = dev->pm;
|
|
|
|
|
|
|
|
if (pm == NULL) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return atomic_test_bit(&pm->flags,
|
|
|
|
PM_DEVICE_FLAG_STATE_LOCKED);
|
|
|
|
}
|
2022-01-06 12:17:38 +01:00
|
|
|
|
|
|
|
bool pm_device_on_power_domain(const struct device *dev)
|
|
|
|
{
|
|
|
|
#ifdef CONFIG_PM_DEVICE_POWER_DOMAIN
|
|
|
|
struct pm_device *pm = dev->pm;
|
|
|
|
|
|
|
|
if (pm == NULL) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return pm->domain != NULL;
|
|
|
|
#else
|
|
|
|
return false;
|
|
|
|
#endif
|
|
|
|
}
|