pm: add device shell commands
Add support for a "pm" shell command to trigger suspend/resume as well as runtime-get/put on devices. This is useful for testing during driver development. Signed-off-by: Fabio Baltieri <fabiobaltieri@google.com>
This commit is contained in:
parent
82db5d3adb
commit
c5003e0eb1
|
@ -7,3 +7,4 @@ endif()
|
|||
|
||||
zephyr_sources_ifdef(CONFIG_PM_DEVICE device.c)
|
||||
zephyr_sources_ifdef(CONFIG_PM_DEVICE_RUNTIME device_runtime.c)
|
||||
zephyr_sources_ifdef(CONFIG_PM_DEVICE_SHELL pm_shell.c)
|
||||
|
|
|
@ -122,6 +122,13 @@ config PM_DEVICE_RUNTIME_EXCLUSIVE
|
|||
On system suspend / resume do not trigger the Device PM hooks but
|
||||
only rely on Runtime PM to manage the devices power states.
|
||||
|
||||
config PM_DEVICE_SHELL
|
||||
bool "Device Power Management shell"
|
||||
depends on SHELL
|
||||
help
|
||||
Enable the device power management shell, for triggering device power
|
||||
management events through the shell interface.
|
||||
|
||||
endif # PM_DEVICE
|
||||
|
||||
endmenu
|
||||
|
|
174
subsys/pm/pm_shell.c
Normal file
174
subsys/pm/pm_shell.c
Normal file
|
@ -0,0 +1,174 @@
|
|||
/*
|
||||
* Copyright 2024 Google LLC
|
||||
*
|
||||
* SPDX-License-Identifier: Apache-2.0
|
||||
*/
|
||||
|
||||
#include <stdio.h>
|
||||
#include <zephyr/shell/shell.h>
|
||||
#include <zephyr/pm/device.h>
|
||||
#include <zephyr/pm/device_runtime.h>
|
||||
|
||||
static void device_name_get(size_t idx, struct shell_static_entry *entry)
|
||||
{
|
||||
const struct device *dev = shell_device_lookup(idx, NULL);
|
||||
|
||||
entry->syntax = (dev != NULL) ? dev->name : NULL;
|
||||
entry->handler = NULL;
|
||||
entry->help = NULL;
|
||||
entry->subcmd = NULL;
|
||||
}
|
||||
|
||||
SHELL_DYNAMIC_CMD_CREATE(dsub_device_name, device_name_get);
|
||||
|
||||
static int pm_cmd_suspend(const struct shell *sh, size_t argc, char *argv[])
|
||||
{
|
||||
const struct device *dev;
|
||||
int ret;
|
||||
|
||||
dev = device_get_binding(argv[1]);
|
||||
if (dev == NULL) {
|
||||
shell_error(sh, "Invalid device: %s", argv[1]);
|
||||
return -ENODEV;
|
||||
}
|
||||
|
||||
if (pm_device_runtime_is_enabled(dev)) {
|
||||
shell_error(sh, "Device %s uses runtime PM, use the runtime functions instead",
|
||||
dev->name);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
ret = pm_device_action_run(dev, PM_DEVICE_ACTION_SUSPEND);
|
||||
if (ret < 0) {
|
||||
shell_error(sh, "Device %s error: %d", "suspend", ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int pm_cmd_resume(const struct shell *sh, size_t argc, char *argv[])
|
||||
{
|
||||
const struct device *dev;
|
||||
int ret;
|
||||
|
||||
dev = device_get_binding(argv[1]);
|
||||
if (dev == NULL) {
|
||||
shell_error(sh, "Invalid device: %s", argv[1]);
|
||||
return -ENODEV;
|
||||
}
|
||||
|
||||
if (pm_device_runtime_is_enabled(dev)) {
|
||||
shell_error(sh, "Device %s uses runtime PM, use the runtime functions instead",
|
||||
dev->name);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
ret = pm_device_action_run(dev, PM_DEVICE_ACTION_RESUME);
|
||||
if (ret < 0) {
|
||||
shell_error(sh, "Device %s error: %d", "resume", ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
#if defined(CONFIG_PM_DEVICE_RUNTIME)
|
||||
static int pm_cmd_runtime_get(const struct shell *sh, size_t argc, char *argv[])
|
||||
{
|
||||
const struct device *dev;
|
||||
int ret;
|
||||
|
||||
dev = device_get_binding(argv[1]);
|
||||
if (dev == NULL) {
|
||||
shell_error(sh, "Invalid device: %s", argv[1]);
|
||||
return -ENODEV;
|
||||
}
|
||||
|
||||
if (!pm_device_runtime_is_enabled(dev)) {
|
||||
shell_error(sh, "Device %s is not using runtime PM", dev->name);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
ret = pm_device_runtime_get(dev);
|
||||
if (ret < 0) {
|
||||
shell_error(sh, "Device %s error: %d", "runtime get", ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int pm_cmd_runtime_put(const struct shell *sh, size_t argc, char *argv[])
|
||||
{
|
||||
const struct device *dev;
|
||||
int ret;
|
||||
|
||||
dev = device_get_binding(argv[1]);
|
||||
if (dev == NULL) {
|
||||
shell_error(sh, "Invalid device: %s", argv[1]);
|
||||
return -ENODEV;
|
||||
}
|
||||
|
||||
if (!pm_device_runtime_is_enabled(dev)) {
|
||||
shell_error(sh, "Device %s is not using runtime PM", dev->name);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
ret = pm_device_runtime_put(dev);
|
||||
if (ret < 0) {
|
||||
shell_error(sh, "Device %s error: %d", "runtime put", ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int pm_cmd_runtime_put_async(const struct shell *sh, size_t argc, char *argv[])
|
||||
{
|
||||
const struct device *dev;
|
||||
int ret;
|
||||
|
||||
dev = device_get_binding(argv[1]);
|
||||
if (dev == NULL) {
|
||||
shell_error(sh, "Invalid device: %s", argv[1]);
|
||||
return -ENODEV;
|
||||
}
|
||||
|
||||
if (!pm_device_runtime_is_enabled(dev)) {
|
||||
shell_error(sh, "Device %s is not using runtime PM", dev->name);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
ret = pm_device_runtime_put_async(dev, K_NO_WAIT);
|
||||
if (ret < 0) {
|
||||
shell_error(sh, "Device %s error: %d", "runtime put async", ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
#endif /* CONFIG_PM_DEVICE_RUNTIME */
|
||||
|
||||
SHELL_STATIC_SUBCMD_SET_CREATE(
|
||||
sub_pm_cmds,
|
||||
SHELL_CMD_ARG(suspend, &dsub_device_name,
|
||||
"Call the PM suspend action on a device",
|
||||
pm_cmd_suspend, 2, 0),
|
||||
SHELL_CMD_ARG(resume, &dsub_device_name,
|
||||
"Call the PM resume action on a device",
|
||||
pm_cmd_resume, 2, 0),
|
||||
#if defined(CONFIG_PM_DEVICE_RUNTIME)
|
||||
SHELL_CMD_ARG(runtime-get, &dsub_device_name,
|
||||
"Call the PM runtime get on a device",
|
||||
pm_cmd_runtime_get, 2, 0),
|
||||
SHELL_CMD_ARG(runtime-put, &dsub_device_name,
|
||||
"Call the PM runtime put on a device",
|
||||
pm_cmd_runtime_put, 2, 0),
|
||||
SHELL_CMD_ARG(runtime-put-async, &dsub_device_name,
|
||||
"Call the PM runtime put async on a device",
|
||||
pm_cmd_runtime_put_async, 2, 0),
|
||||
#endif /* CONFIG_PM_DEVICE_RUNTIME */
|
||||
SHELL_SUBCMD_SET_END);
|
||||
|
||||
SHELL_CMD_REGISTER(pm, &sub_pm_cmds, "PM commands", NULL);
|
Loading…
Reference in a new issue