/*
task.cpp - Task Switching Library for AVR-G++
Copyright (c) 2021 Sasapea's Lab. All right reserved.
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Lesser General Public
License as published by the Free Software Foundation; either
version 2.1 of the License, or (at your option) any later version.
This library is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General
Public License along with this library; if not, write to the
Free Software Foundation, Inc., 59 Temple Place, Suite 330,
Boston, MA 02111-1307 USA
*/
#include <stdlib.h>
#include "task.h"
#if defined(ARDUINO)
#include "Arduino.h"
#endif
void *Task::_head;
void *Task::_tail;
#define ASM(...) __asm__ volatile (__VA_ARGS__)
#define PUSHA() \
ASM("push r29"); \
ASM("push r28"); \
ASM("push r17"); \
ASM("push r16"); \
ASM("push r15"); \
ASM("push r14"); \
ASM("push r13"); \
ASM("push r12"); \
ASM("push r11"); \
ASM("push r10"); \
ASM("push r9"); \
ASM("push r8"); \
ASM("push r7"); \
ASM("push r6"); \
ASM("push r5"); \
ASM("push r4"); \
ASM("push r3"); \
ASM("push r2"); \
ASM("push __tmp_reg__")
#define POPA() \
ASM("pop __tmp_reg__"); \
ASM("pop r2"); \
ASM("pop r3"); \
ASM("pop r4"); \
ASM("pop r5"); \
ASM("pop r6"); \
ASM("pop r7"); \
ASM("pop r8"); \
ASM("pop r9"); \
ASM("pop r10"); \
ASM("pop r11"); \
ASM("pop r12"); \
ASM("pop r13"); \
ASM("pop r14"); \
ASM("pop r15"); \
ASM("pop r16"); \
ASM("pop r17"); \
ASM("pop r28"); \
ASM("pop r29")
//
// Number of cycles from function call to return
//
// 123 cycles (7.6875us/16MHz)
//
void Task::yield(void) // (3)
{
//
// save context (39)
//
ASM("in __tmp_reg__, __SREG__");
PUSHA();
//
// check tail link (7)
//
ASM("lds r28, %[TAIL] + 0" :: [TAIL] "m" (_tail));
ASM("lds r29, %[TAIL] + 1" :: [TAIL] "m" (_tail));
ASM("adiw r28, 0");
ASM("breq 9f"); // --> no task
//
// add tail link (14)
//
ASM("push r1");
ASM("push r1");
ASM("in r24, __SP_L__");
ASM("in r25, __SP_H__");
ASM("std y + 1, r24");
ASM("std y + 2, r25");
ASM("sts %[TAIL] + 0, r24" :: [TAIL] "m" (_tail));
ASM("sts %[TAIL] + 1, r25" :: [TAIL] "m" (_tail));
//
// change stack pointer (10)
//
ASM("lds r28, %[HEAD] + 0" :: [HEAD] "m" (_head));
ASM("lds r29, %[HEAD] + 1" :: [HEAD] "m" (_head));
ASM("ldd __tmp_reg__, y + 3"); // load __SREG__ from stack
#if defined(__AVR_XMEGA__) && __AVR_XMEGA__
ASM("out __SREG__, __tmp_reg__");
ASM("out __SP_L__, r28");
ASM("out __SP_H__, r29");
#else
ASM("cli");
ASM("out __SP_L__, r28");
ASM("out __SREG__, __tmp_reg__");
ASM("out __SP_H__, r29");
#endif
//
// update head link (8)
//
ASM("pop r24");
ASM("pop r25");
ASM("sts %[HEAD] + 0, r24" :: [HEAD] "m" (_head));
ASM("sts %[HEAD] + 1, r25" :: [HEAD] "m" (_head));
//
// restore context (42)
//
ASM("9:");
POPA();
ASM("ret");
}
bool Task::start(uint16_t size, void (&func)(void))
{
//
// allocate new task stack area
//
register uint8_t *stack = (uint8_t *)malloc(size);
if (stack == 0)
return false;
stack += size - 1;
//
// save stack pointer
//
ASM("in r20, __SP_L__" ::: "r20");
ASM("in r21, __SP_H__" ::: "r21");
//
// change stack pointer
//
ASM("in __tmp_reg__, __SREG__");
#if defined(__AVR_XMEGA__) && __AVR_XMEGA__
ASM("out __SP_L__, %A[STACK]" :: [STACK] "r" (stack));
ASM("out __SP_H__, %B[STACK]" :: [STACK] "r" (stack));
#else
ASM("cli");
ASM("out __SP_L__, %A[STACK]" :: [STACK] "r" (stack));
ASM("out __SREG__, __tmp_reg__");
ASM("out __SP_H__, %B[STACK]" :: [STACK] "r" (stack));
#endif
//
// push jump address
//
ASM("push %A[FUNC]" :: [FUNC] "r" (func));
ASM("push %B[FUNC]" :: [FUNC] "r" (func));
#if defined(__AVR_3_BYTE_PC__) && __AVR_3_BYTE_PC__
ASM("push r1"); // zero
#endif
//
// save context
//
PUSHA();
//
// add tail link
//
ASM("push r1");
ASM("push r1");
ASM("lds r28, %[TAIL] + 0" :: [TAIL] "m" (_tail));
ASM("lds r29, %[TAIL] + 1" :: [TAIL] "m" (_tail));
ASM("adiw r28, 0");
ASM("brne 1f");
ASM("ldi r28, lo8(%[HEAD] - 1)" :: [HEAD] "m" (_head));
ASM("ldi r29, hi8(%[HEAD] - 1)" :: [HEAD] "m" (_head));
ASM("1:");
ASM("in r24, __SP_L__");
ASM("in r25, __SP_H__");
ASM("std y + 1, r24");
ASM("std y + 2, r25");
ASM("sts %[TAIL] + 0, r24" :: [TAIL] "m" (_tail));
ASM("sts %[TAIL] + 1, r25" :: [TAIL] "m" (_tail));
//
// restore stack pointer
//
#if defined(__AVR_XMEGA__) && __AVR_XMEGA__
ASM("out __SP_L__, r20");
ASM("out __SP_H__, r21");
#else
ASM("cli");
ASM("out __SP_L__, r20");
ASM("out __SREG__, __tmp_reg__");
ASM("out __SP_H__, r21");
#endif
return true;
}
#if defined(ARDUINO)
void Task::delay(uint32_t ms)
{
uint32_t t = millis();
while(millis() - t < ms)
yield();
}
void Task::delayMicroseconds(uint32_t us)
{
uint32_t t = micros();
while(micros() - t < us)
yield();
}
#endif