version 1.8, 2000/09/23 15:46:59
|
version 1.11, 2003/03/09 15:16:57
|
Line 1
|
Line 1
|
\ Multitasker 19aug94py |
\ Multitasker 19aug94py |
|
|
\ Copyright (C) 1995,1996,1997 Free Software Foundation, Inc. |
\ Copyright (C) 1995,1996,1997,2001 Free Software Foundation, Inc. |
|
|
\ This file is part of Gforth. |
\ This file is part of Gforth. |
|
|
Line 20
|
Line 20
|
|
|
Create sleepers sleepers A, sleepers A, 0 , |
Create sleepers sleepers A, sleepers A, 0 , |
|
|
\ LINK-TASK links task1 into the task chain of task2 |
|
: link-task ( task1 task2 -- ) |
: link-task ( task1 task2 -- ) |
over 2@ 2dup cell+ ! swap ! \ unlink task1 |
\G LINK-TASK links task1 into the task chain of task2 |
2dup @ cell+ ! 2dup dup @ rot 2! ! ; |
over 2@ 2dup cell+ ! swap ! \ unlink task1 |
|
2dup @ cell+ ! 2dup dup @ rot 2! ! ; |
|
|
|
: sleep ( task -- ) |
|
\G deactivates task |
|
sleepers link-task ; |
|
: wake ( task -- ) |
|
\G activates task |
|
next-task link-task ; |
|
|
: sleep ( task -- ) sleepers link-task ; |
|
: wake ( task -- ) next-task link-task ; |
|
|
|
\ PAUSE is the task-switcher |
|
: pause ( -- ) |
: pause ( -- ) |
rp@ fp@ lp@ sp@ save-task ! |
\G PAUSE is the task-switcher |
next-task @ up! save-task @ sp! |
rp@ fp@ lp@ sp@ save-task ! |
lp! fp! rp! ; |
next-task @ up! save-task @ sp! |
|
lp! fp! rp! ; |
|
|
\ STOP sleeps a task and switches to the next |
|
: stop ( -- ) |
: stop ( -- ) |
rp@ fp@ lp@ sp@ save-task ! |
\G STOP sleeps a task and switches to the next |
next-task @ up! save-task @ sp! |
rp@ fp@ lp@ sp@ save-task ! |
lp! fp! rp! prev-task @ sleep ; |
next-task @ up! save-task @ sp! |
|
lp! fp! rp! prev-task @ sleep ; |
|
|
\ USER' computes the task offset |
|
:noname ' >body @ ; |
:noname ' >body @ ; |
:noname ' >body @ postpone literal ; |
:noname ' >body @ postpone literal ; |
interpret/compile: user' ( 'user' -- n ) |
interpret/compile: user' ( 'user' -- n ) |
|
\G USER' computes the task offset of a user variable |
|
|
\ NEWTASK creates a new, sleeping task |
: NewTask ( stacksize -- Task ) dup 2* 2* udp @ + dup |
: NewTask ( n -- Task ) dup 2* 2* udp @ + dup |
\G NEWTASK creates a new, sleeping task |
allocate throw + >r |
allocate throw + >r |
r@ over - udp @ - next-task over udp @ move |
r@ over - udp @ - next-task over udp @ move |
r> over user' rp0 + ! dup >r |
r> over user' rp0 + ! dup >r |
dup r@ user' lp0 + ! over - |
dup r@ user' lp0 + ! over - |
dup r@ user' fp0 + ! over - |
dup r@ user' fp0 + ! over - |
dup r@ user' sp0 + ! over - |
dup r@ user' sp0 + ! over - |
dup r@ user' normal-dp + dup >r ! |
dup r@ user' normal-dp + dup >r ! |
r> r@ user' dpp + ! + $10 + |
r> r@ user' dpp + ! 2drop |
r@ user' >tib + ! |
0 r@ user' current-input + ! |
r> dup 2dup 2! dup sleep ; |
r> dup 2dup 2! dup sleep ; |
|
|
: kill-task |
Create killer killer A, killer A, |
next-task @ up! save-task @ sp! |
: kill ( task -- ) |
lp! fp! rp! prev-task @ dup dup link-task user' normal-dp + @ free throw ; |
\G kills a task - deactivate and free task area |
|
dup killer link-task killer dup dup 2! |
|
user' normal-dp + @ free throw ; |
|
|
|
: kill-task ( -- ) |
|
\G kills the current task, also on bottom of return stack of a new task |
|
next-task @ up! save-task @ sp! |
|
lp! fp! rp! prev-task @ kill ; |
|
|
: (pass) ( x1 .. xn n task -- ) rdrop |
: (pass) ( x1 .. xn n task -- ) rdrop |
[ ' kill-task >body ] ALiteral r> |
[ ' kill-task >body ] ALiteral r> |
Line 71 interpret/compile: user' ( 'user' -- n )
|
Line 82 interpret/compile: user' ( 'user' -- n )
|
cells r@ user' sp0 + @ tuck swap - dup r@ user' save-task + ! |
cells r@ user' sp0 + @ tuck swap - dup r@ user' save-task + ! |
?DO I ! cell +LOOP r> wake ; |
?DO I ! cell +LOOP r> wake ; |
|
|
: activate ( task -- ) 0 swap (pass) ; |
: activate ( task -- ) |
: pass ( x1 .. xn n task -- ) (pass) ; |
\G activates the task. |
|
\G Continues execution with the caller of ACTIVATE. |
|
0 swap (pass) ; |
|
: pass ( x1 .. xn n task -- ) |
|
\G passes n parameters to the task and activates that task. |
|
\G Continues execution with the caller of PASS. |
|
(pass) ; |
|
|
: single-tasking? ( -- flag ) |
: single-tasking? ( -- flag ) |
|
\G checks if only one task is running |
next-task dup @ = ; |
next-task dup @ = ; |
|
|
: task-key BEGIN pause key? single-tasking? or UNTIL (key) ; |
: task-key BEGIN pause key? single-tasking? or UNTIL (key) ; |