2018-05-14 23:55:15 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2014 Freie Universität Berlin, Hinnerk van Bruinehsen
|
|
|
|
* 2017 RWTH Aachen, Josua Arndt
|
|
|
|
* 2018 Matthew Blue
|
2023-06-27 22:09:21 +02:00
|
|
|
* 2021-2023 Gerson Fernando Budke
|
2018-05-14 23:55:15 +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
|
2018-05-14 23:55:15 +02:00
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Implementation of the CPU initialization
|
|
|
|
*
|
|
|
|
* @author Hinnerk van Bruinehsen <h.v.bruinehsen@fu-berlin.de>
|
|
|
|
* @author Steffen Robertz <steffen.robertz@rwth-aachen.de>
|
|
|
|
* @author Josua Arndt <jarndt@ias.rwth-aachen.de>
|
|
|
|
* @author Matthew Blue <matthew.blue.neuro@gmail.com>
|
|
|
|
* @author Francisco Acosta <francisco.acosta@inria.fr>
|
2021-01-02 18:40:07 +01:00
|
|
|
* @author Gerson Fernando Budke <nandojve@gmail.com>
|
|
|
|
*
|
2018-05-14 23:55:15 +02:00
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <avr/io.h>
|
|
|
|
#include <avr/wdt.h>
|
|
|
|
#include <avr/pgmspace.h>
|
|
|
|
|
|
|
|
#include "cpu.h"
|
2021-02-04 02:03:29 +01:00
|
|
|
#include "cpu_clock.h"
|
2018-05-14 23:55:15 +02:00
|
|
|
#include "board.h"
|
2021-02-04 02:09:58 +01:00
|
|
|
#include "irq.h"
|
2018-05-14 23:55:15 +02:00
|
|
|
#include "periph/init.h"
|
|
|
|
#include "panic.h"
|
2021-02-04 02:03:29 +01:00
|
|
|
#include "kernel_defines.h"
|
2018-05-14 23:55:15 +02:00
|
|
|
|
2020-10-22 11:34:00 +02:00
|
|
|
#define ENABLE_DEBUG 0
|
2018-05-14 23:55:15 +02:00
|
|
|
#include "debug.h"
|
|
|
|
|
2021-01-02 19:22:11 +01:00
|
|
|
#ifdef RST
|
|
|
|
/* In ATxmega there is a special register to get reset cause */
|
|
|
|
#define MCUSR RST.STATUS
|
|
|
|
#else
|
2020-02-09 09:40:26 +01:00
|
|
|
#ifndef MCUSR
|
|
|
|
/* In older ATmegas the MCUSR register was still named MCUCSR. Current avrlibc
|
|
|
|
* versions provide the MCUSR macro for those as well, but adding a fallback
|
|
|
|
* here doesn't hurt*/
|
|
|
|
#define MCUSR MCUCSR
|
|
|
|
#endif /* !MCUSR */
|
2021-01-02 19:22:11 +01:00
|
|
|
#endif /* RST */
|
2020-02-09 09:40:26 +01:00
|
|
|
|
2018-05-14 23:55:15 +02:00
|
|
|
/*
|
|
|
|
* Since atmega MCUs do not feature a software reset, the watchdog timer
|
|
|
|
* is being used. It will be set to the shortest time and then force a
|
2019-10-23 21:13:51 +02:00
|
|
|
* reset. Therefore the MCUSR register needs to be reset as fast as
|
2018-05-14 23:55:15 +02:00
|
|
|
* possible.
|
|
|
|
* Which means in the bootloader or in the following init0 if no bootloader is used.
|
|
|
|
* Bootloader resets watchdog and pass MCUSR in r2 (e.g. Optiboot) in order to pass
|
|
|
|
* information about the reset cause to the application.
|
|
|
|
* When no Bootloader is used the watchdog will be disabled in the init0 section.
|
|
|
|
* When a software reset was triggered, r3 will contain 0xAA.
|
|
|
|
* In order to prevent changes to the values from the .init section, MCUSR and r3
|
|
|
|
* are saved in the .init0 section
|
|
|
|
*/
|
|
|
|
uint8_t mcusr_mirror __attribute__((section(".noinit")));
|
|
|
|
uint8_t soft_rst __attribute__((section(".noinit")));
|
2023-06-27 22:41:18 +02:00
|
|
|
#if (AVR8_STATE_IRQ_USE_SRAM)
|
|
|
|
uint8_t avr8_state_irq_count_sram = 0;
|
|
|
|
#endif
|
2023-06-27 22:09:21 +02:00
|
|
|
#if (AVR8_STATE_UART_USE_SRAM)
|
|
|
|
uint8_t avr8_state_uart_sram = 0;
|
|
|
|
#endif
|
2021-01-02 18:40:07 +01:00
|
|
|
|
2020-02-09 09:40:26 +01:00
|
|
|
void get_mcusr(void) __attribute__((naked, section(".init0"), used));
|
2018-05-14 23:55:15 +02:00
|
|
|
|
|
|
|
void get_mcusr(void)
|
|
|
|
{
|
|
|
|
/* save soft reset flag set in reset routine */
|
|
|
|
__asm__ __volatile__("mov %0, r3\n" : "=r" (soft_rst) :);
|
|
|
|
#ifdef BOOTLOADER_CLEARS_WATCHDOG_AND_PASSES_MCUSR
|
|
|
|
/* save the reset flags passed from the bootloader */
|
|
|
|
__asm__ __volatile__("mov %0, r2\n" : "=r" (mcusr_mirror) :);
|
|
|
|
#else
|
|
|
|
/* save the reset flags */
|
2020-02-09 09:40:26 +01:00
|
|
|
mcusr_mirror = MCUSR;
|
|
|
|
MCUSR = 0;
|
2018-05-14 23:55:15 +02:00
|
|
|
wdt_disable();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void cpu_init(void)
|
|
|
|
{
|
2021-08-27 17:06:50 +02:00
|
|
|
#ifdef PRUSB
|
|
|
|
/* disable usb interrupt */
|
|
|
|
PRR1 |= 1<<PRUSB;
|
|
|
|
#endif
|
|
|
|
|
2021-01-02 18:40:07 +01:00
|
|
|
avr8_reset_cause();
|
2018-05-14 23:55:15 +02:00
|
|
|
|
|
|
|
wdt_reset(); /* should not be nececessary as done in bootloader */
|
|
|
|
wdt_disable(); /* but when used without bootloader this is needed */
|
|
|
|
|
2021-02-04 02:03:29 +01:00
|
|
|
avr8_clk_init();
|
|
|
|
|
2019-04-10 10:18:43 +02:00
|
|
|
/* Initialize stdio before periph_init() to allow use of DEBUG() there */
|
|
|
|
#ifdef MODULE_AVR_LIBC_EXTRA
|
2021-01-02 18:40:07 +01:00
|
|
|
avr8_stdio_init();
|
2019-04-10 10:18:43 +02:00
|
|
|
#endif
|
2018-05-14 23:55:15 +02:00
|
|
|
/* Initialize peripherals for which modules are included in the makefile.*/
|
|
|
|
/* spi_init */
|
|
|
|
/* rtc_init */
|
|
|
|
/* hwrng_init */
|
|
|
|
periph_init();
|
2021-02-04 02:09:58 +01:00
|
|
|
|
2021-01-12 00:02:48 +01:00
|
|
|
#ifdef CPU_ATXMEGA
|
|
|
|
/* Enable Multilevel Interrupt Controller */
|
|
|
|
PMIC.CTRL |= PMIC_HILVLEN_bm
|
|
|
|
| PMIC_MEDLVLEN_bm
|
|
|
|
| PMIC_LOLVLEN_bm;
|
|
|
|
#endif
|
|
|
|
|
2023-06-27 22:09:21 +02:00
|
|
|
/* Set global resources initial state */
|
|
|
|
avr8_state_uart = 0;
|
2023-06-27 22:41:18 +02:00
|
|
|
avr8_state_irq_count = 0;
|
2023-06-27 22:09:21 +02:00
|
|
|
|
2021-02-04 02:09:58 +01:00
|
|
|
irq_enable();
|
2018-05-14 23:55:15 +02:00
|
|
|
}
|
|
|
|
|
2019-02-03 18:59:27 +01:00
|
|
|
struct __freelist {
|
|
|
|
size_t size;
|
|
|
|
struct __freelist *next;
|
|
|
|
};
|
|
|
|
|
|
|
|
extern struct __freelist *__flp;
|
|
|
|
extern char *__malloc_heap_start;
|
|
|
|
extern char *__malloc_heap_end;
|
|
|
|
extern char *__brkval;
|
|
|
|
|
|
|
|
void heap_stats(void)
|
|
|
|
{
|
|
|
|
int heap_size = __malloc_heap_end - __malloc_heap_start;
|
|
|
|
int free = __malloc_heap_end - __brkval;
|
|
|
|
struct __freelist *fp;
|
|
|
|
for (fp = __flp; fp; fp = fp->next) {
|
|
|
|
free += fp->size;
|
|
|
|
}
|
|
|
|
printf("heap: %d (used %d, free %d) [bytes]\n",
|
|
|
|
heap_size, heap_size - free, free);
|
|
|
|
}
|