mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
150 lines
3.2 KiB
C
150 lines
3.2 KiB
C
/*
|
||
* Copyright (C) 2021 Gerson Fernando Budke <nandojve@gmail.com>
|
||
*
|
||
* 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 cpu_atxmega
|
||
* @ingroup cpu_atxmega_periph
|
||
* @{
|
||
*
|
||
* @file
|
||
* @brief Low-level PM driver implementation
|
||
*
|
||
* @author Gerson Fernando Budke <nandojve@gmail.com>
|
||
*
|
||
* @}
|
||
*/
|
||
|
||
#include <avr/sleep.h>
|
||
|
||
#include "cpu_pm.h"
|
||
#include "irq.h"
|
||
#include "periph/pm.h"
|
||
#include "periph_conf.h"
|
||
|
||
#define ENABLE_DEBUG 0
|
||
#include "debug.h"
|
||
|
||
#define PWR_REG_BASE ((uint16_t)&PR)
|
||
#define PWR_REG_OFFSET (0x01)
|
||
|
||
/**
|
||
* @brief Extract the device id of the given power reduction mask
|
||
*/
|
||
static inline uint8_t _device_mask(pwr_reduction_t pwr)
|
||
{
|
||
return (pwr & 0xff);
|
||
}
|
||
|
||
/**
|
||
* @brief Extract the register id of the given power reduction mask
|
||
*/
|
||
static inline uint8_t _register_id(pwr_reduction_t pwr)
|
||
{
|
||
return (pwr >> 8) & 0xff;
|
||
}
|
||
|
||
/**
|
||
* @brief Generate the register index of the given power reduction mask
|
||
*/
|
||
static inline uint8_t *_register_addr(pwr_reduction_t pwr)
|
||
{
|
||
uint8_t id = _register_id(pwr);
|
||
uint16_t addr = PWR_REG_BASE + (id * PWR_REG_OFFSET);
|
||
|
||
return (uint8_t *)addr;
|
||
}
|
||
|
||
void pm_reboot(void)
|
||
{
|
||
DEBUG("Reboot Software Reset\n" );
|
||
|
||
/* XMEGA AU [MANUAL] p. 116 CTRL->Control register
|
||
* page 13 3.12.1 Sequence for write operation to protected I/O registers
|
||
* page 15 3.14.1 CCP – Configuration Change Protection register
|
||
*/
|
||
|
||
/* Disable CCP for Protected IO registerand set new value*/
|
||
_PROTECTED_WRITE(RST_CTRL, RST_SWRST_bm);
|
||
while (1) {}
|
||
}
|
||
|
||
/*
|
||
* DEBUG may affect this routine.
|
||
*
|
||
* --- Do NOT add DEBUG macro here ---
|
||
*
|
||
*/
|
||
void pm_set(unsigned mode)
|
||
{
|
||
unsigned irq_state = irq_disable();
|
||
|
||
if (avr8_is_uart_tx_pending() && mode < 4) {
|
||
irq_restore(irq_state);
|
||
return;
|
||
}
|
||
|
||
switch (mode) {
|
||
case 0:
|
||
set_sleep_mode(SLEEP_SMODE_PDOWN_gc);
|
||
break;
|
||
case 1:
|
||
set_sleep_mode(SLEEP_SMODE_PSAVE_gc);
|
||
break;
|
||
case 2:
|
||
set_sleep_mode(SLEEP_SMODE_STDBY_gc);
|
||
break;
|
||
case 3:
|
||
set_sleep_mode(SLEEP_SMODE_ESTDBY_gc);
|
||
break;
|
||
default:
|
||
set_sleep_mode(SLEEP_SMODE_IDLE_gc);
|
||
}
|
||
sleep_enable();
|
||
sei();
|
||
sleep_cpu();
|
||
sleep_disable();
|
||
irq_restore(irq_state);
|
||
}
|
||
|
||
void pm_periph_enable(pwr_reduction_t pwr)
|
||
{
|
||
uint8_t mask = _device_mask(pwr);
|
||
uint8_t *reg = _register_addr(pwr);
|
||
|
||
*reg &= ~mask;
|
||
}
|
||
|
||
void pm_periph_disable(pwr_reduction_t pwr)
|
||
{
|
||
uint8_t mask = _device_mask(pwr);
|
||
uint8_t *reg = _register_addr(pwr);
|
||
|
||
*reg |= mask;
|
||
}
|
||
|
||
void pm_periph_power_off(void)
|
||
{
|
||
uint8_t *reg = _register_addr(PWR_GENERAL_POWER);
|
||
uint8_t i;
|
||
|
||
/* Freeze all peripheral clocks */
|
||
for (i = 0; i <= 7; i++) {
|
||
reg[i] = 0xff;
|
||
}
|
||
|
||
/* EBI Must be always enabled when configured */
|
||
#if defined (__AVR_ATxmega64A1__) || \
|
||
defined (__AVR_ATxmega64A1U__) || \
|
||
defined (__AVR_ATxmega128A1__) || \
|
||
defined (__AVR_ATxmega128A1U__)
|
||
if (ebi_config.addr_bits > 0) {
|
||
reg[0] &= ~PR_EBI_bm;
|
||
}
|
||
#endif
|
||
}
|