-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathCoroutines.Mod
76 lines (58 loc) · 1.82 KB
/
Coroutines.Mod
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
(* OBERON System 3, Release 2.3.
Copyright 1998 Institute of Computer Systems of ETH Zürich, ETH Center, CH-8092 Zürich,
e-mail: [email protected].
This module may be used under the conditions of the general Oberon
System 3 license contract. The full text can be downloaded from
"ftp://ftp.inf.ethz.ch/pub/software/Oberon/System3/license.txt;A"
Under the license terms stated it is in particular (a) prohibited to modify
the interface of this module in any way that disagrees with the style
or content of the system and (b) requested to provide all conversions
of the source code to another platform with the name OBERON. *)
MODULE Coroutines; (** non-portable *)
(* NW 3. 8. 89/JG 22. 5. 92/ARD 2. 9. 92 *)
(* ARD 6. 10. 93, Oberon version, compile with /no *)
(* pjm 14.11.96, port to Native Oberon *)
(* eos 13.8.98, added Init *)
IMPORT SYSTEM;
TYPE Coroutine* = POINTER TO CorDesc;
Proc* = PROCEDURE(me: Coroutine);
Stack* = POINTER TO RECORD END;
CorDesc* = RECORD
SP: LONGINT; P*: Proc; S*: Stack; L*: LONGINT;
END;
VAR
main*: Coroutine; (** main Coroutine *)
cur: Coroutine;
(** Initialising a Coroutine c:
NEW(c); Init(c, Handler, stackSize); *)
PROCEDURE Init*(me: Coroutine; proc: Proc; size: LONGINT);
BEGIN
me.P := proc; me.L := size; SYSTEM.NEW(me.S, size)
END Init;
(** Start - Transfer control to a coroutine for the first time *)
PROCEDURE Start*(me: Coroutine);
(*CODE {SYSTEM.i386}
MOV ECX, cur
MOV [ECX], EBP
MOV EAX, me[EBP]
MOV EBX, 8[EAX]
ADD EBX, 12[EAX]
MOV ESP, EBX
PUSH EAX
MOV cur, EAX
CALL DWORD 4[EAX]
PUSH 29
INT 3*)
END Start;
(** Transfer - Transfer control to a coroutine *)
PROCEDURE Transfer*(to: Coroutine);
(*CODE {SYSTEM.i386}
MOV ECX, cur
MOV [ECX], EBP
MOV ECX, to[EBP]
MOV EBP, [ECX]
MOV cur, ECX*)
END Transfer;
BEGIN
NEW(main); cur := main
END Coroutines.