esp8266: implement task based scheduler
I have chosed to call this implementation `esp8266` instead of `xtensa` as it has been written specifically for the ESP8266 and there are no other Xtensa chips with the CALL0 ABI (no windowing) that I know of. The only other related chip is the ESP32, which does implement register windowing and thus needs a very different implementation.
Этот коммит содержится в:
родитель
caf35cfc41
коммит
098f900363
3 изменённых файлов: 128 добавлений и 1 удалений
54
src/internal/task/task_stack_esp8266.S
Обычный файл
54
src/internal/task/task_stack_esp8266.S
Обычный файл
|
@ -0,0 +1,54 @@
|
||||||
|
.section .text.tinygo_startTask,"ax",@progbits
|
||||||
|
.global tinygo_startTask
|
||||||
|
.type tinygo_startTask, %function
|
||||||
|
tinygo_startTask:
|
||||||
|
// Small assembly stub for starting a goroutine. This is already run on the
|
||||||
|
// new stack, with the callee-saved registers already loaded.
|
||||||
|
// Most importantly, r4 contains the pc of the to-be-started function and r5
|
||||||
|
// contains the only argument it is given. Multiple arguments are packed
|
||||||
|
// into one by storing them in a new allocation.
|
||||||
|
|
||||||
|
// Set the first argument of the goroutine start wrapper, which contains all
|
||||||
|
// the arguments.
|
||||||
|
mov.n a2, a13
|
||||||
|
|
||||||
|
// Branch to the "goroutine start" function.
|
||||||
|
callx0 a12
|
||||||
|
|
||||||
|
// After return, exit this goroutine. This is a tail call.
|
||||||
|
call0 tinygo_pause
|
||||||
|
.size tinygo_startTask, .-tinygo_startTask
|
||||||
|
|
||||||
|
.global tinygo_swapTask
|
||||||
|
.type tinygo_swapTask, %function
|
||||||
|
tinygo_swapTask:
|
||||||
|
// This function gets the following parameters:
|
||||||
|
// a2 = newStack uintptr
|
||||||
|
// a3 = oldStack *uintptr
|
||||||
|
// Note:
|
||||||
|
// a0 is the return address
|
||||||
|
// a1 is the stack pointer (sp)
|
||||||
|
|
||||||
|
// Save all callee-saved registers:
|
||||||
|
addi sp, sp, -20
|
||||||
|
s32i.n a12, sp, 0
|
||||||
|
s32i.n a13, sp, 4
|
||||||
|
s32i.n a14, sp, 8
|
||||||
|
s32i.n a15, sp, 12
|
||||||
|
s32i.n a0, sp, 16
|
||||||
|
|
||||||
|
// Save the current stack pointer in oldStack.
|
||||||
|
s32i.n sp, a3, 0
|
||||||
|
|
||||||
|
// Switch to the new stack pointer.
|
||||||
|
mov.n sp, a2
|
||||||
|
|
||||||
|
// Load state from new task and branch to the previous position in the
|
||||||
|
// program.
|
||||||
|
l32i.n a12, sp, 0
|
||||||
|
l32i.n a13, sp, 4
|
||||||
|
l32i.n a14, sp, 8
|
||||||
|
l32i.n a15, sp, 12
|
||||||
|
l32i.n a0, sp, 16
|
||||||
|
addi sp, sp, 20
|
||||||
|
ret.n
|
70
src/internal/task/task_stack_esp8266.go
Обычный файл
70
src/internal/task/task_stack_esp8266.go
Обычный файл
|
@ -0,0 +1,70 @@
|
||||||
|
// +build scheduler.tasks,esp8266
|
||||||
|
|
||||||
|
package task
|
||||||
|
|
||||||
|
// Stack switch implementation for the ESP8266, which does not use the windowed
|
||||||
|
// ABI of Xtensa. Registers are assigned as follows:
|
||||||
|
// a0: return address (link register)
|
||||||
|
// a1: stack pointer (must be 16-byte aligned)
|
||||||
|
// a2-a7: incoming arguments
|
||||||
|
// a8: static chain (unused)
|
||||||
|
// a12-a15: callee-saved
|
||||||
|
// a15: stack frame pointer (optional, unused)
|
||||||
|
// Sources:
|
||||||
|
// http://cholla.mmto.org/esp8266/xtensa.html
|
||||||
|
// https://0x04.net/~mwk/doc/xtensa.pdf
|
||||||
|
|
||||||
|
import "unsafe"
|
||||||
|
|
||||||
|
var systemStack uintptr
|
||||||
|
|
||||||
|
// calleeSavedRegs is the list of registers that must be saved and restored when
|
||||||
|
// switching between tasks. Also see task_stack_esp8266.S that relies on the
|
||||||
|
// exact layout of this struct.
|
||||||
|
type calleeSavedRegs struct {
|
||||||
|
a12 uintptr
|
||||||
|
a13 uintptr
|
||||||
|
a14 uintptr
|
||||||
|
a15 uintptr
|
||||||
|
|
||||||
|
pc uintptr // also link register or r0
|
||||||
|
}
|
||||||
|
|
||||||
|
// archInit runs architecture-specific setup for the goroutine startup.
|
||||||
|
func (s *state) archInit(r *calleeSavedRegs, fn uintptr, args unsafe.Pointer) {
|
||||||
|
// Store the initial sp for the startTask function (implemented in assembly).
|
||||||
|
s.sp = uintptr(unsafe.Pointer(r))
|
||||||
|
|
||||||
|
// Initialize the registers.
|
||||||
|
// These will be popped off of the stack on the first resume of the goroutine.
|
||||||
|
|
||||||
|
// Start the function at tinygo_startTask (defined in
|
||||||
|
// src/internal/task/task_stack_esp8266.S).
|
||||||
|
// This assembly code calls a function (passed in a12) with a single argument
|
||||||
|
// (passed in a13). After the function returns, it calls Pause().
|
||||||
|
r.pc = uintptr(unsafe.Pointer(&startTask))
|
||||||
|
|
||||||
|
// Pass the function to call in a12.
|
||||||
|
// This function is a compiler-generated wrapper which loads arguments out of a struct pointer.
|
||||||
|
// See createGoroutineStartWrapper (defined in compiler/goroutine.go) for more information.
|
||||||
|
r.a12 = fn
|
||||||
|
|
||||||
|
// Pass the pointer to the arguments struct in a13.
|
||||||
|
r.a13 = uintptr(args)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *state) resume() {
|
||||||
|
swapTask(s.sp, &systemStack)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *state) pause() {
|
||||||
|
newStack := systemStack
|
||||||
|
systemStack = 0
|
||||||
|
swapTask(newStack, &s.sp)
|
||||||
|
}
|
||||||
|
|
||||||
|
// SystemStack returns the system stack pointer when called from a task stack.
|
||||||
|
// When called from the system stack, it returns 0.
|
||||||
|
func SystemStack() uintptr {
|
||||||
|
return systemStack
|
||||||
|
}
|
|
@ -2,13 +2,16 @@
|
||||||
"inherits": ["xtensa"],
|
"inherits": ["xtensa"],
|
||||||
"cpu": "esp8266",
|
"cpu": "esp8266",
|
||||||
"build-tags": ["esp8266", "esp"],
|
"build-tags": ["esp8266", "esp"],
|
||||||
|
"scheduler": "tasks",
|
||||||
"linker": "xtensa-esp32-elf-ld",
|
"linker": "xtensa-esp32-elf-ld",
|
||||||
|
"default-stack-size": 2048,
|
||||||
"cflags": [
|
"cflags": [
|
||||||
"-mcpu=esp8266"
|
"-mcpu=esp8266"
|
||||||
],
|
],
|
||||||
"linkerscript": "targets/esp8266.ld",
|
"linkerscript": "targets/esp8266.ld",
|
||||||
"extra-files": [
|
"extra-files": [
|
||||||
"src/device/esp/esp8266.S"
|
"src/device/esp/esp8266.S",
|
||||||
|
"src/internal/task/task_stack_esp8266.S"
|
||||||
],
|
],
|
||||||
"binary-format": "esp8266",
|
"binary-format": "esp8266",
|
||||||
"flash-command": "esptool.py --chip=esp8266 --port {port} write_flash 0x00000 {bin} -fm qio"
|
"flash-command": "esptool.py --chip=esp8266 --port {port} write_flash 0x00000 {bin} -fm qio"
|
||||||
|
|
Загрузка…
Создание таблицы
Сослаться в новой задаче