/* * Copyright (C) 2016 Kaspar Schleiser * * 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_pm_layered * @{ * * @file * @brief Platform-independent power management code * * @author Kaspar Schleiser * * @} */ #include #include "board.h" #include "atomic_utils.h" #include "irq.h" #include "periph/pm.h" #include "pm_layered.h" #define ENABLE_DEBUG 0 #include "debug.h" #ifndef PM_NUM_MODES #error PM_NUM_MODES must be defined in periph_cpu.h! #endif #ifndef PM_BLOCKER_INITIAL #define PM_BLOCKER_INITIAL 0x01010101 /* block all by default */ #endif /** * @brief Global variable for keeping track of blocked modes */ static pm_blocker_t pm_blocker = { .val_u32 = PM_BLOCKER_INITIAL }; void pm_set_lowest(void) { pm_blocker_t blocker = { .val_u32 = atomic_load_u32(&pm_blocker.val_u32) }; unsigned mode = PM_NUM_MODES; while (mode) { if (blocker.val_u8[mode-1]) { break; } mode--; } /* set lowest mode if blocker is still the same */ unsigned state = irq_disable(); if (blocker.val_u32 == pm_blocker.val_u32) { DEBUG("pm: setting mode %u\n", mode); pm_set(mode); } else { DEBUG("pm: mode block changed\n"); } irq_restore(state); } void pm_block(unsigned mode) { assert(pm_blocker.val_u8[mode] != 255); unsigned state = irq_disable(); pm_blocker.val_u8[mode]++; irq_restore(state); } void pm_unblock(unsigned mode) { assert(pm_blocker.val_u8[mode] > 0); unsigned state = irq_disable(); pm_blocker.val_u8[mode]--; irq_restore(state); } pm_blocker_t pm_get_blocker(void) { pm_blocker_t result = { .val_u32 = atomic_load_u32(&pm_blocker.val_u32) }; return result; } #ifndef PROVIDES_PM_LAYERED_OFF void pm_off(void) { irq_disable(); while(1) { pm_set(0); } } #endif