mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-15 14:52:45 +01:00
345 lines
11 KiB
C
345 lines
11 KiB
C
/*
|
|
* Copyright (C) 2014 Freie Universität Berlin, Hinnerk van Bruinehsen
|
|
*
|
|
* 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_atmega_common
|
|
* @{
|
|
*
|
|
* @file thread_arch.c
|
|
* @brief Implementation of the kernel's architecture dependent thread interface
|
|
*
|
|
* @author Hinnerk van Bruinehsen <h.v.bruinehsen@fu-berlin.de>
|
|
*
|
|
* @}
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
|
|
#include "arch/thread_arch.h"
|
|
#include "thread.h"
|
|
#include "sched.h"
|
|
#include "irq.h"
|
|
#include "cpu.h"
|
|
#include "kernel_internal.h"
|
|
|
|
/*
|
|
* local function declarations (prefixed with __)
|
|
*/
|
|
|
|
static void __context_save(void);
|
|
static void __context_restore(void);
|
|
static void __enter_thread_mode(void);
|
|
|
|
/**
|
|
* @brief Since AVR doesn't support direct manipulation of the program counter we
|
|
* model a stack like it would be left by __context_save().
|
|
* The resulting layout in memory is the following:
|
|
* ---------------tcb_t (not created by thread_arch_stack_init) ----------
|
|
* local variables (a temporary value and the stackpointer)
|
|
* -----------------------------------------------------------------------
|
|
* a marker (AFFE) - for debugging purposes (helps finding the stack
|
|
* -----------------------------------------------------------------------
|
|
* a 16 Bit pointer to sched_task_exit
|
|
* (Optional 17 bit (bit is set to zero) for ATmega2560
|
|
* -----------------------------------------------------------------------
|
|
* a 16 Bit pointer to task_func
|
|
* this is placed exactly at the place where the program counter would be
|
|
* stored normally and thus can be returned to when __context_restore()
|
|
* has been run
|
|
* (Optional 17 bit (bit is set to zero) for ATmega2560
|
|
* -----------------------------------------------------------------------
|
|
* saved registers from context:
|
|
* r0
|
|
* status register
|
|
* (Optional EIND and RAMPZ registers for ATmega2560)
|
|
* r1 - r23
|
|
* pointer to arg in r24 and r25
|
|
* r26 - r31
|
|
* -----------------------------------------------------------------------
|
|
*
|
|
* After the invocation of __context_restore() the pointer to task_func is
|
|
* on top of the stack and can be returned to. This way we can actually place
|
|
* it inside of the programm counter of the MCU.
|
|
* if task_func returns sched_task_exit gets popped into the PC
|
|
*/
|
|
char *thread_arch_stack_init(thread_task_func_t task_func, void *arg,
|
|
void *stack_start, int stack_size)
|
|
{
|
|
uint16_t tmp_adress;
|
|
uint8_t *stk;
|
|
|
|
/* AVR uses 16 Bit or two 8 Bit registers for storing pointers*/
|
|
stk = (uint8_t *)((uintptr_t)stack_start + stack_size);
|
|
|
|
/* put marker on stack */
|
|
stk--;
|
|
*stk = (uint8_t) 0xAF;
|
|
stk--;
|
|
*stk = (uint8_t) 0xFE;
|
|
|
|
/* save sched_task_exit */
|
|
stk--;
|
|
tmp_adress = (uint16_t) sched_task_exit;
|
|
*stk = (uint8_t)(tmp_adress & (uint16_t) 0x00ff);
|
|
stk--;
|
|
tmp_adress >>= 8;
|
|
*stk = (uint8_t)(tmp_adress & (uint16_t) 0x00ff);
|
|
|
|
#if defined(__AVR_ATmega2560__)
|
|
/* The ATMega2560 uses a 17 bit PC, we set whole the top byte forcibly to 0 */
|
|
stk--;
|
|
*stk = (uint8_t) 0x00;
|
|
#endif
|
|
|
|
/* save address to task_func in place of the program counter */
|
|
stk--;
|
|
tmp_adress = (uint16_t) task_func;
|
|
*stk = (uint8_t)(tmp_adress & (uint16_t) 0x00ff);
|
|
stk--;
|
|
tmp_adress >>= 8;
|
|
*stk = (uint8_t)(tmp_adress & (uint16_t) 0x00ff);
|
|
|
|
#if defined(__AVR_ATmega2560__)
|
|
/* The ATMega2560 uses a 17 byte PC, we set the top byte forcibly to 0 */
|
|
stk--;
|
|
*stk = (uint8_t) 0x00;
|
|
#endif
|
|
|
|
|
|
/* r0 */
|
|
stk--;
|
|
*stk = (uint8_t) 0x00;
|
|
|
|
/* status register (with interrupts enabled) */
|
|
stk--;
|
|
*stk = (uint8_t) 0x80;
|
|
|
|
#if defined(__AVR_ATmega2560__)
|
|
/* EIND */
|
|
stk--;
|
|
*stk = (uint8_t) 0x00;
|
|
|
|
/* RAMPZ */
|
|
stk--;
|
|
*stk = (uint8_t) 0x00;
|
|
#endif
|
|
|
|
/* r1 - has always to be 0 */
|
|
stk--;
|
|
*stk = (uint8_t) 0x00;
|
|
/*
|
|
* Space for registers r2 -r23
|
|
*
|
|
* use loop for better readability, the compiler unrolls anyways
|
|
*/
|
|
|
|
int i;
|
|
|
|
for (i = 2; i <= 23 ; i++) {
|
|
stk--;
|
|
*stk = (uint8_t) 0;
|
|
}
|
|
|
|
/*
|
|
* In accordance with the AVR calling conventions *arg has to be inside
|
|
* r24 and r25
|
|
* */
|
|
stk--;
|
|
tmp_adress = (uint16_t) arg;
|
|
*stk = (uint8_t)(tmp_adress & (uint16_t) 0x00ff);
|
|
stk--;
|
|
tmp_adress >>= 8;
|
|
*stk = (uint8_t)(tmp_adress & (uint16_t) 0x00ff);
|
|
|
|
/*
|
|
* Space for registers r26-r31
|
|
*/
|
|
for (i = 26; i <= 31; i++) {
|
|
stk--;
|
|
*stk = (uint8_t) i;
|
|
}
|
|
|
|
stk--;
|
|
return (char *) stk;
|
|
}
|
|
|
|
/**
|
|
* @brief thread_arch_stack_print prints the stack to stdout.
|
|
* It depends on getting the correct values for stack_start, stack_size and sp
|
|
* from sched_active_thread.
|
|
* Maybe it would be good to change that to way that is less dependant on
|
|
* getting correct values elsewhere (since it is a debugging tool and in the
|
|
* presence of bugs the data may be corrupted).
|
|
*/
|
|
void thread_arch_stack_print(void)
|
|
{
|
|
uint8_t found_marker = 1;
|
|
uint8_t *sp = (uint8_t *)sched_active_thread->sp;
|
|
uint16_t size = 0;
|
|
|
|
printf("Printing current stack of thread %" PRIkernel_pid "\n", thread_getpid());
|
|
printf("\taddress:\tdata:\n");
|
|
|
|
do {
|
|
printf("\t0x%04x:\t\t0x%04x\n", (unsigned int)sp, (unsigned int)*sp);
|
|
sp++;
|
|
size++;
|
|
|
|
if ((*sp == 0xFE) && (*(sp + 1) == 0xAF)) {
|
|
found_marker = 0;
|
|
}
|
|
}
|
|
while (found_marker == 1);
|
|
|
|
printf("stack size: %u bytes\n", size);
|
|
}
|
|
|
|
void thread_arch_start_threading(void) __attribute__((naked));
|
|
void thread_arch_start_threading(void)
|
|
{
|
|
sched_run();
|
|
__enter_thread_mode();
|
|
}
|
|
|
|
/**
|
|
* @brief Set the MCU into Thread-Mode and load the initial task from the stack and run it
|
|
*/
|
|
|
|
void NORETURN __enter_thread_mode(void) __attribute__((naked));
|
|
void NORETURN __enter_thread_mode(void)
|
|
{
|
|
enableIRQ();
|
|
__context_restore();
|
|
asm volatile("ret");
|
|
|
|
UNREACHABLE();
|
|
}
|
|
|
|
void thread_arch_yield(void) __attribute__((naked));
|
|
void thread_arch_yield(void)
|
|
{
|
|
__context_save();
|
|
|
|
/* disableIRQ(); */ /* gets already disabled during __context_save() */
|
|
sched_run();
|
|
enableIRQ();
|
|
|
|
__context_restore();
|
|
asm volatile("ret");
|
|
}
|
|
|
|
|
|
__attribute__((always_inline)) static inline void __context_save(void)
|
|
{
|
|
asm volatile(
|
|
"push r0 \n\t"
|
|
"in r0, __SREG__ \n\t"
|
|
"cli \n\t"
|
|
"push r0 \n\t"
|
|
#if defined(__AVR_ATmega2560__)
|
|
/* EIND and RAMPZ */
|
|
"in r0, 0x3b \n\t"
|
|
"push r0 \n\t"
|
|
"in r0, 0x3c \n\t"
|
|
"push r0 \n\t"
|
|
#endif
|
|
"push r1 \n\t"
|
|
"clr r1 \n\t"
|
|
"push r2 \n\t"
|
|
"push r3 \n\t"
|
|
"push r4 \n\t"
|
|
"push r5 \n\t"
|
|
"push r6 \n\t"
|
|
"push r7 \n\t"
|
|
"push r8 \n\t"
|
|
"push r9 \n\t"
|
|
"push r10 \n\t"
|
|
"push r11 \n\t"
|
|
"push r12 \n\t"
|
|
"push r13 \n\t"
|
|
"push r14 \n\t"
|
|
"push r15 \n\t"
|
|
"push r16 \n\t"
|
|
"push r17 \n\t"
|
|
"push r18 \n\t"
|
|
"push r19 \n\t"
|
|
"push r20 \n\t"
|
|
"push r21 \n\t"
|
|
"push r22 \n\t"
|
|
"push r23 \n\t"
|
|
"push r24 \n\t"
|
|
"push r25 \n\t"
|
|
"push r26 \n\t"
|
|
"push r27 \n\t"
|
|
"push r28 \n\t"
|
|
"push r29 \n\t"
|
|
"push r30 \n\t"
|
|
"push r31 \n\t"
|
|
"lds r26, sched_active_thread \n\t"
|
|
"lds r27, sched_active_thread + 1 \n\t"
|
|
"in r0, __SP_L__ \n\t"
|
|
"st x+, r0 \n\t"
|
|
"in r0, __SP_H__ \n\t"
|
|
"st x+, r0 \n\t"
|
|
);
|
|
|
|
}
|
|
|
|
__attribute__((always_inline)) static inline void __context_restore(void)
|
|
{
|
|
asm volatile(
|
|
"lds r26, sched_active_thread \n\t"
|
|
"lds r27, sched_active_thread + 1 \n\t"
|
|
"ld r28, x+ \n\t"
|
|
"out __SP_L__, r28 \n\t"
|
|
"ld r29, x+ \n\t"
|
|
"out __SP_H__, r29 \n\t"
|
|
"pop r31 \n\t"
|
|
"pop r30 \n\t"
|
|
"pop r29 \n\t"
|
|
"pop r28 \n\t"
|
|
"pop r27 \n\t"
|
|
"pop r26 \n\t"
|
|
"pop r25 \n\t"
|
|
"pop r24 \n\t"
|
|
"pop r23 \n\t"
|
|
"pop r22 \n\t"
|
|
"pop r21 \n\t"
|
|
"pop r20 \n\t"
|
|
"pop r19 \n\t"
|
|
"pop r18 \n\t"
|
|
"pop r17 \n\t"
|
|
"pop r16 \n\t"
|
|
"pop r15 \n\t"
|
|
"pop r14 \n\t"
|
|
"pop r13 \n\t"
|
|
"pop r12 \n\t"
|
|
"pop r11 \n\t"
|
|
"pop r10 \n\t"
|
|
"pop r9 \n\t"
|
|
"pop r8 \n\t"
|
|
"pop r7 \n\t"
|
|
"pop r6 \n\t"
|
|
"pop r5 \n\t"
|
|
"pop r4 \n\t"
|
|
"pop r3 \n\t"
|
|
"pop r2 \n\t"
|
|
"pop r1 \n\t"
|
|
#if defined(__AVR_ATmega2560__)
|
|
/* EIND and RAMPZ */
|
|
"pop r0 \n\t"
|
|
"out 0x3c, r0 \n\t"
|
|
"pop r0 \n\t"
|
|
"out 0x3b, r0 \n\t"
|
|
#endif
|
|
"pop r0 \n\t"
|
|
"out __SREG__, r0 \n\t"
|
|
"pop r0 \n\t"
|
|
);
|
|
}
|