2019-07-03 17:44:03 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2019 Thomas Stilwell <stilwellt@openlabs.co>
|
|
|
|
*
|
|
|
|
* 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
|
|
|
|
*
|
2019-07-10 03:38:37 +02:00
|
|
|
* @author Bas Stottelaar <basstottelaar@gmail.com>
|
|
|
|
* @author Vincent Dupont <vincent@otakeys.com>
|
2019-07-03 17:44:03 +02:00
|
|
|
* @author Thomas Stilwell <stilwellt@openlabs.co>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
2019-07-10 03:38:37 +02:00
|
|
|
#include <stdlib.h>
|
2019-07-03 17:44:03 +02:00
|
|
|
#include <string.h>
|
2019-07-10 03:38:37 +02:00
|
|
|
#include "periph/pm.h"
|
2019-07-03 17:44:03 +02:00
|
|
|
|
2019-07-10 03:38:37 +02:00
|
|
|
#ifdef MODULE_PM_LAYERED
|
|
|
|
#include "pm_layered.h"
|
2019-07-03 17:44:03 +02:00
|
|
|
|
2019-07-10 03:38:37 +02:00
|
|
|
#endif /* MODULE_PM_LAYERED */
|
2019-07-03 17:44:03 +02:00
|
|
|
|
|
|
|
static void _print_usage(void) {
|
2019-07-10 03:38:37 +02:00
|
|
|
puts("Usage:");
|
|
|
|
#ifdef MODULE_PM_LAYERED
|
|
|
|
puts("\tpm show: display current blockers for each power mode");
|
|
|
|
puts("\tpm set <mode>: manually set power mode (lasts until WFI returns)");
|
|
|
|
puts("\tpm block <mode>: manually block power mode");
|
|
|
|
puts("\tpm unblock <mode>: 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 <power mode>\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;
|
2019-07-03 17:44:03 +02:00
|
|
|
}
|
|
|
|
|
2019-07-10 03:38:37 +02:00
|
|
|
static int cmd_set(char *arg)
|
2019-07-03 17:44:03 +02:00
|
|
|
{
|
2019-07-10 03:38:37 +02:00
|
|
|
int mode = parse_mode(arg);
|
|
|
|
if (mode < 0) {
|
2019-07-03 17:44:03 +02:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2019-07-10 03:38:37 +02:00
|
|
|
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;
|
2019-07-03 17:44:03 +02:00
|
|
|
}
|
|
|
|
|
2020-04-29 11:07:12 +02:00
|
|
|
pm_blocker_t pm_blocker = pm_get_blocker();
|
2019-07-10 03:38:37 +02:00
|
|
|
if (pm_blocker.val_u8[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;
|
2019-07-03 17:44:03 +02:00
|
|
|
uint8_t lowest_allowed_mode = 0;
|
|
|
|
|
2020-04-29 11:07:12 +02:00
|
|
|
pm_blocker_t pm_blocker = pm_get_blocker();
|
2019-07-03 17:44:03 +02:00
|
|
|
for (unsigned i = 0; i < PM_NUM_MODES; i++) {
|
|
|
|
printf("mode %u blockers: %u \n", i, pm_blocker.val_u8[i]);
|
2019-07-10 03:38:37 +02:00
|
|
|
if (pm_blocker.val_u8[i]) {
|
2019-07-03 17:44:03 +02:00
|
|
|
lowest_allowed_mode = i + 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-10 03:38:37 +02:00
|
|
|
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();
|
2019-07-03 17:44:03 +02:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2019-07-10 03:38:37 +02:00
|
|
|
|
|
|
|
int _pm_handler(int argc, char **argv)
|
|
|
|
{
|
2020-05-02 21:55:28 +02:00
|
|
|
if (argc < 2) {
|
|
|
|
_print_usage();
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2019-07-10 03:38:37 +02:00
|
|
|
#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;
|
|
|
|
}
|