/* * Copyright (C) 2019 Thomas Stilwell * * This file is subject to the terms and conditions of the GNU Lesser * General Public License v2.1. See the file LICENSE in the top level * directory for more details. */ /** * @ingroup sys_shell_commands * @{ * * @file * @brief Shell command to interact with the PM subsystem * * @author Bas Stottelaar * @author Vincent Dupont * @author Thomas Stilwell * * @} */ #include #include #include #include "periph/pm.h" #include "shell.h" #ifdef MODULE_PM_LAYERED #include "pm_layered.h" #endif /* MODULE_PM_LAYERED */ static void _print_usage(void) { puts("Usage:"); #ifdef MODULE_PM_LAYERED puts("\tpm show: display current blockers for each power mode"); puts("\tpm set : manually set power mode (lasts until WFI returns)"); puts("\tpm block : manually block power mode"); puts("\tpm unblock : manually unblock power mode"); #endif /* MODULE_PM_LAYERED */ puts("\tpm off: call pm_off()"); } #ifdef MODULE_PM_LAYERED static int check_mode(int argc, char **argv) { if (argc != 3) { printf("Usage: %s %s \n", argv[0], argv[1]); return -1; } return 0; } static int parse_mode(char *argv) { uint8_t mode = atoi(argv); if (mode >= PM_NUM_MODES) { printf("Error: power mode not in range 0 - %d.\n", PM_NUM_MODES - 1); return -1; } return mode; } static int cmd_block(char *arg) { int mode = parse_mode(arg); if (mode < 0) { return 1; } printf("Blocking power mode %d.\n", mode); fflush(stdout); pm_block(mode); return 0; } static int cmd_set(char *arg) { int mode = parse_mode(arg); if (mode < 0) { return 1; } printf("CPU is entering power mode %d.\n", mode); puts("Now waiting for a wakeup event..."); fflush(stdout); pm_set(mode); /* execution stops here until anything (like shell input) wakes the CPU */ printf("CPU has returned from power mode %d.\n", mode); return 0; } static int cmd_unblock(char *arg) { int mode = parse_mode(arg); if (mode < 0) { return 1; } pm_blocker_t pm_blocker = pm_get_blocker(); if (pm_blocker.blockers[mode] == 0) { printf("Mode %d is already unblocked.\n", mode); return 1; } printf("Unblocking power mode %d.\n", mode); fflush(stdout); pm_unblock(mode); return 0; } static int cmd_show(char *arg) { (void)arg; uint8_t lowest_allowed_mode = 0; pm_blocker_t pm_blocker = pm_get_blocker(); for (unsigned i = 0; i < PM_NUM_MODES; i++) { printf("mode %u blockers: %u \n", i, pm_blocker.blockers[i]); if (pm_blocker.blockers[i]) { lowest_allowed_mode = i + 1; } } printf("Lowest allowed mode: %u\n", lowest_allowed_mode); return 0; } #endif /* MODULE_PM_LAYERED */ static int cmd_off(char *arg) { (void)arg; pm_off(); return 0; } static int _pm_handler(int argc, char **argv) { if (argc < 2) { _print_usage(); return 1; } #ifdef MODULE_PM_LAYERED if (!strcmp(argv[1], "show")) { if (argc != 2) { puts("usage: pm show: display current blockers for each power mode"); return 1; } return cmd_show(argv[1]); } if (!strcmp(argv[1], "block")) { if (check_mode(argc, argv) != 0) { return 1; } return cmd_block(argv[2]); } if (!strcmp(argv[1], "unblock")) { if (check_mode(argc, argv) != 0) { return 1; } return cmd_unblock(argv[2]); } if (!strcmp(argv[1], "set")) { if (check_mode(argc, argv) != 0) { return 1; } return cmd_set(argv[2]); } #endif /* MODULE_PM_LAYERED */ if (!strcmp(argv[1], "off")) { return cmd_off(NULL); } _print_usage(); return 1; } SHELL_COMMAND(pm, "interact with layered PM subsystem", _pm_handler);