2014-07-15 12:08:52 +02:00
|
|
|
/*
|
2015-09-14 00:27:27 +02:00
|
|
|
* Copyright (C) 2015 Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
* 2014 Freie Universität Berlin, Hinnerk van Bruinehsen
|
2018-03-05 19:42:41 +01:00
|
|
|
* 2018 RWTH Aachen, Josua Arndt <jarndt@ias.rwth-aachen.de>
|
2023-06-27 22:09:21 +02:00
|
|
|
* 2021-2023 Gerson Fernando Budke <nandojve@gmail.com>
|
2014-07-15 12:08:52 +02:00
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2021-01-02 18:40:07 +01:00
|
|
|
* @ingroup cpu_avr8_common
|
|
|
|
* @brief Common implementations and headers for AVR-8 family based micro-controllers
|
2014-07-15 12:08:52 +02:00
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
2021-01-02 18:40:07 +01:00
|
|
|
* @brief Basic definitions for the AVR-8 common module
|
2014-07-15 12:08:52 +02:00
|
|
|
*
|
|
|
|
* When ever you want to do something hardware related, that is accessing MCUs registers directly,
|
|
|
|
* just include this file. It will then make sure that the MCU specific headers are included.
|
|
|
|
*
|
|
|
|
* @author Stefan Pfeiffer <stefan.pfeiffer@fu-berlin.de>
|
|
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
|
|
* @author Hinnerk van Bruinehsen <h.v.bruinehsen@fu-berlin.de>
|
2015-09-14 00:27:27 +02:00
|
|
|
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
2018-03-05 19:42:41 +01:00
|
|
|
* @author Josua Arndt <jarndt@ias.rwth-aachen.de>
|
2021-01-02 18:40:07 +01:00
|
|
|
* @author Gerson Fernando Budke <nandojve@gmail.com>
|
2018-03-05 19:42:41 +01:00
|
|
|
*
|
2014-07-15 12:08:52 +02:00
|
|
|
*/
|
|
|
|
|
2017-05-23 18:19:52 +02:00
|
|
|
#ifndef CPU_H
|
|
|
|
#define CPU_H
|
2014-07-15 12:08:52 +02:00
|
|
|
|
2015-09-04 16:51:08 +02:00
|
|
|
#include <stdio.h>
|
2015-09-14 00:27:27 +02:00
|
|
|
#include <stdint.h>
|
2014-07-15 12:08:52 +02:00
|
|
|
|
2015-09-14 00:27:27 +02:00
|
|
|
#include <avr/interrupt.h>
|
2022-06-17 17:19:49 +02:00
|
|
|
#include "architecture.h"
|
2015-09-04 16:51:08 +02:00
|
|
|
#include "cpu_conf.h"
|
2021-01-02 15:30:51 +01:00
|
|
|
#include "cpu_clock.h"
|
2018-04-25 17:53:26 +02:00
|
|
|
#include "sched.h"
|
|
|
|
#include "thread.h"
|
2023-06-27 22:09:21 +02:00
|
|
|
#include "states_internal.h"
|
2014-10-13 10:53:20 +02:00
|
|
|
|
|
|
|
#ifdef __cplusplus
|
2018-06-08 17:52:44 +02:00
|
|
|
extern "C"
|
|
|
|
{
|
2014-10-13 10:53:20 +02:00
|
|
|
#endif
|
|
|
|
|
2022-10-20 18:50:22 +02:00
|
|
|
/**
|
|
|
|
* @name BOD monitoring when CPU is on sleep
|
|
|
|
* @{
|
|
|
|
*/
|
|
|
|
#ifndef AVR8_PM_DISABLE_BOD_ON_SLEEP
|
|
|
|
#define AVR8_PM_DISABLE_BOD_ON_SLEEP 0 /**< BOD is active on sleep */
|
|
|
|
#endif
|
|
|
|
/** @} */
|
|
|
|
|
2024-01-09 23:07:20 +01:00
|
|
|
/**
|
|
|
|
* @brief CPU cycles per busy wait loop
|
|
|
|
*/
|
|
|
|
#define CPU_CYCLES_PER_LOOP (7)
|
|
|
|
|
2018-05-30 21:34:22 +02:00
|
|
|
/**
|
|
|
|
* @name Use shared I2C functions
|
|
|
|
* @{
|
|
|
|
*/
|
|
|
|
#define PERIPH_I2C_NEED_READ_REG
|
|
|
|
#define PERIPH_I2C_NEED_WRITE_REG
|
|
|
|
#define PERIPH_I2C_NEED_READ_REGS
|
|
|
|
#define PERIPH_I2C_NEED_WRITE_REGS
|
|
|
|
/** @} */
|
|
|
|
|
2020-02-06 12:21:48 +01:00
|
|
|
/**
|
|
|
|
* @brief Run this code on entering interrupt routines
|
2015-09-14 00:27:27 +02:00
|
|
|
*/
|
2021-01-02 18:40:07 +01:00
|
|
|
static inline void avr8_enter_isr(void)
|
2015-09-14 00:27:27 +02:00
|
|
|
{
|
2020-02-06 12:21:48 +01:00
|
|
|
/* This flag is only called from IRQ context, and nested IRQs are not
|
|
|
|
* supported as of now. The flag will be unset before the IRQ context is
|
|
|
|
* left, so no need to use memory barriers or atomics here
|
|
|
|
*/
|
2023-06-27 22:41:18 +02:00
|
|
|
++avr8_state_irq_count;
|
2020-02-06 12:21:48 +01:00
|
|
|
}
|
|
|
|
|
2023-06-27 22:09:21 +02:00
|
|
|
/**
|
|
|
|
* @brief Compute UART TX channel
|
|
|
|
*
|
|
|
|
* @param uart The UART number
|
|
|
|
*/
|
|
|
|
#define AVR8_STATE_FLAG_UART_TX(uart) (0x01U << uart)
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Set UART TX channel as pending
|
|
|
|
*
|
|
|
|
* @param uart The UART number
|
|
|
|
*/
|
|
|
|
static inline void avr8_uart_tx_set_pending(unsigned uart)
|
|
|
|
{
|
|
|
|
avr8_state_uart |= AVR8_STATE_FLAG_UART_TX(uart);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Clear UART TX channel pending state
|
|
|
|
*
|
|
|
|
* @param uart The UART number
|
|
|
|
*/
|
|
|
|
static inline void avr8_uart_tx_clear_pending(unsigned uart)
|
|
|
|
{
|
|
|
|
avr8_state_uart &= ~AVR8_STATE_FLAG_UART_TX(uart);
|
|
|
|
}
|
|
|
|
|
2020-02-06 12:21:48 +01:00
|
|
|
/**
|
|
|
|
* @brief Check if TX on any present UART device is still pending
|
|
|
|
*
|
|
|
|
* @retval !=0 At least on UART device is still sending data out
|
|
|
|
* @retval 0 No UART is currently sending data
|
|
|
|
*/
|
2021-01-02 18:40:07 +01:00
|
|
|
static inline int avr8_is_uart_tx_pending(void)
|
2020-02-06 12:21:48 +01:00
|
|
|
{
|
2023-06-27 22:09:21 +02:00
|
|
|
return avr8_state_uart;
|
2015-09-14 00:27:27 +02:00
|
|
|
}
|
|
|
|
|
2018-04-25 17:53:26 +02:00
|
|
|
/**
|
|
|
|
* @brief Run this code on exiting interrupt routines
|
2015-09-14 00:27:27 +02:00
|
|
|
*/
|
2021-01-02 18:40:07 +01:00
|
|
|
void avr8_exit_isr(void);
|
2015-09-14 00:27:27 +02:00
|
|
|
|
2021-02-04 02:03:29 +01:00
|
|
|
/**
|
|
|
|
* @brief Initialization of the CPU clock
|
|
|
|
*/
|
|
|
|
void avr8_clk_init(void);
|
|
|
|
|
2015-09-04 16:51:08 +02:00
|
|
|
/**
|
2022-06-17 17:19:49 +02:00
|
|
|
* @brief Get the last instruction's address
|
2022-06-20 19:56:00 +02:00
|
|
|
*
|
|
|
|
* @details This works only if called in a function as first statement, as
|
|
|
|
* it relies on the return address to be the topmost item on the stack.
|
2015-09-04 16:51:08 +02:00
|
|
|
*/
|
2022-06-17 17:19:49 +02:00
|
|
|
static inline uinttxtptr_t __attribute__((always_inline)) cpu_get_caller_pc(void)
|
2015-09-04 16:51:08 +02:00
|
|
|
{
|
2022-06-17 17:19:49 +02:00
|
|
|
uinttxtptr_t addr;
|
2022-06-20 19:56:00 +02:00
|
|
|
__asm__ volatile(
|
|
|
|
"ldi %D[dest], 0" "\n\t"
|
|
|
|
#if __AVR_3_BYTE_PC__
|
|
|
|
"pop %C[dest] " "\n\t"
|
|
|
|
#else
|
|
|
|
"ldi %C[dest], 0" "\n\t"
|
|
|
|
#endif
|
|
|
|
"pop %B[dest]" "\n\t"
|
|
|
|
"pop %A[dest]" "\n\t"
|
|
|
|
"push %A[dest]" "\n\t"
|
|
|
|
"push %B[dest]" "\n\t"
|
|
|
|
#if __AVR_3_BYTE_PC__
|
|
|
|
"push %C[dest] " "\n\t"
|
|
|
|
#endif
|
|
|
|
: [dest] "=r"(addr)
|
2021-09-30 10:39:21 +02:00
|
|
|
: /* no inputs */
|
2022-06-20 19:56:00 +02:00
|
|
|
: "memory"
|
2021-09-30 10:39:21 +02:00
|
|
|
);
|
2022-06-20 19:56:00 +02:00
|
|
|
|
|
|
|
/* addr now contains instruction to return to, subtract one to get
|
|
|
|
* the instruction that called this function. Also multiply by two to get
|
|
|
|
* the byte position, rather than the (16 bit) instruction position */
|
|
|
|
addr = (addr - 1 ) * 2;
|
2022-06-17 17:19:49 +02:00
|
|
|
return addr;
|
2015-09-04 16:51:08 +02:00
|
|
|
}
|
|
|
|
|
2018-03-26 21:46:12 +02:00
|
|
|
/**
|
|
|
|
* @brief Initializes avrlibc stdio
|
|
|
|
*/
|
2021-01-02 18:40:07 +01:00
|
|
|
void avr8_stdio_init(void);
|
2018-03-26 21:46:12 +02:00
|
|
|
|
2021-01-02 15:36:29 +01:00
|
|
|
/**
|
|
|
|
* @brief Print reset cause
|
|
|
|
*/
|
2021-01-02 18:40:07 +01:00
|
|
|
void avr8_reset_cause(void);
|
2021-01-02 15:36:29 +01:00
|
|
|
|
2014-10-13 10:53:20 +02:00
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
2014-07-15 12:08:52 +02:00
|
|
|
|
2017-05-23 18:19:52 +02:00
|
|
|
#endif /* CPU_H */
|
2014-07-15 12:08:52 +02:00
|
|
|
/** @} */
|