| /* |
| * 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; |
| } |
| |
| if (pm_device_state_is_locked(dev)) { |
| return -EPERM; |
| } |
| |
| /* 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); |
| } |
| |
| void pm_device_state_lock(const struct device *dev) |
| { |
| struct pm_device_base *pm = dev->pm_base; |
| |
| if ((pm != NULL) && !pm_device_runtime_is_enabled(dev)) { |
| atomic_set_bit(&pm->flags, PM_DEVICE_FLAG_STATE_LOCKED); |
| } |
| } |
| |
| void pm_device_state_unlock(const struct device *dev) |
| { |
| struct pm_device_base *pm = dev->pm_base; |
| |
| 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_base *pm = dev->pm_base; |
| |
| if (pm == NULL) { |
| return false; |
| } |
| |
| return atomic_test_bit(&pm->flags, |
| PM_DEVICE_FLAG_STATE_LOCKED); |
| } |
| |
| 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; |
| } |