R-PAGE
Resistance's Portable-Adventure-Game-Engine
coroutine.h
Go to the documentation of this file.
1 /* coroutine.h
2  *
3  * Coroutine mechanics, implemented on top of standard ANSI C. See
4  * https://www.chiark.greenend.org.uk/~sgtatham/coroutines.html for
5  * a full discussion of the theory behind this.
6  *
7  * To use these macros to define a coroutine, you need to write a
8  * function that looks something like this.
9  *
10  * [Simple version using static variables (scr macros)]
11  * int ascending (void) {
12  * static int i;
13  *
14  * scrBegin;
15  * for (i=0; i<10; i++) {
16  * scrReturn(i);
17  * }
18  * scrFinish(-1);
19  * }
20  *
21  * [Re-entrant version using an explicit context structure (ccr macros)]
22  * int ascending (ccrContParam) {
23  * ccrBeginContext;
24  * int i;
25  * ccrEndContext(foo);
26  *
27  * ccrBegin(foo);
28  * for (foo->i=0; foo->i<10; foo->i++) {
29  * ccrReturn(foo->i);
30  * }
31  * ccrFinish(-1);
32  * }
33  *
34  * In the static version, you need only surround the function body
35  * with `scrBegin' and `scrFinish', and then you can do `scrReturn'
36  * within the function and on the next call control will resume
37  * just after the scrReturn statement. Any local variables you need
38  * to be persistent across an `scrReturn' must be declared static.
39  *
40  * In the re-entrant version, you need to declare your persistent
41  * variables between `ccrBeginContext' and `ccrEndContext'. These
42  * will be members of a structure whose name you specify in the
43  * parameter to `ccrEndContext'.
44  *
45  * The re-entrant macros will malloc() the state structure on first
46  * call, and free() it when `ccrFinish' is reached. If you want to
47  * abort in the middle, you can use `ccrStop' to free the state
48  * structure immediately (equivalent to an explicit return() in a
49  * caller-type routine).
50  *
51  * A coroutine returning void type may call `ccrReturnV',
52  * `ccrFinishV' and `ccrStopV', or `scrReturnV', to avoid having to
53  * specify an empty parameter to the ordinary return macros.
54  *
55  * Ground rules:
56  * - never put `ccrReturn' or `scrReturn' within an explicit `switch'.
57  * - never put two `ccrReturn' or `scrReturn' statements on the same
58  * source line.
59  *
60  * The caller of a static coroutine calls it just as if it were an
61  * ordinary function:
62  *
63  * void main(void) {
64  * int i;
65  * do {
66  * i = ascending();
67  * printf("got number %d\n", i);
68  * } while (i != -1);
69  * }
70  *
71  * The caller of a re-entrant coroutine must provide a context
72  * variable:
73  *
74  * void main(void) {
75  * ccrContext z = 0;
76  * do {
77  * printf("got number %d\n", ascending (&z));
78  * } while (z);
79  * }
80  *
81  * Note that the context variable is set back to zero when the
82  * coroutine terminates (by crStop, or by control reaching
83  * crFinish). This can make the re-entrant coroutines more useful
84  * than the static ones, because you can tell when they have
85  * finished.
86  *
87  * If you need to dispose of a crContext when it is non-zero (that
88  * is, if you want to stop calling a coroutine without suffering a
89  * memory leak), the caller should call `ccrAbort(ctx)' where `ctx'
90  * is the context variable.
91  *
92  * This mechanism could have been better implemented using GNU C
93  * and its ability to store pointers to labels, but sadly this is
94  * not part of the ANSI C standard and so the mechanism is done by
95  * case statements instead. That's why you can't put a crReturn()
96  * inside a switch() statement.
97  */
98 
99 /*
100  * coroutine.h is copyright 1995,2000 Simon Tatham.
101  *
102  * Permission is hereby granted, free of charge, to any person
103  * obtaining a copy of this software and associated documentation
104  * files (the "Software"), to deal in the Software without
105  * restriction, including without limitation the rights to use,
106  * copy, modify, merge, publish, distribute, sublicense, and/or
107  * sell copies of the Software, and to permit persons to whom the
108  * Software is furnished to do so, subject to the following
109  * conditions:
110  *
111  * The above copyright notice and this permission notice shall be
112  * included in all copies or substantial portions of the Software.
113  *
114  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
115  * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
116  * OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
117  * NONINFRINGEMENT. IN NO EVENT SHALL SIMON TATHAM BE LIABLE FOR
118  * ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF
119  * CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
120  * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
121  * SOFTWARE.
122  *
123  * $Id$
124  */
125 
126 #ifndef COROUTINE_H
127 #define COROUTINE_H
128 
129 #include <stdlib.h>
130 
131 /*
132  * `scr' macros for static coroutines.
133  */
134 
135 #define scrBegin static int scrLine = 0; switch(scrLine) { case 0:;
136 #define scrFinish(z) } return (z)
137 #define scrFinishV } return
138 
139 #define scrReturn(z) \
140  do {\
141  scrLine=__LINE__;\
142  return (z); case __LINE__:;\
143  } while (0)
144 #define scrReturnV \
145  do {\
146  scrLine=__LINE__;\
147  return; case __LINE__:;\
148  } while (0)
149 
150 /*
151  * `ccr' macros for re-entrant coroutines.
152  */
153 
154 #define ccrContParam void **ccrParam
155 
156 #define ccrBeginContext struct ccrContextTag { int ccrLine
157 #define ccrEndContext(x) } *x = (struct ccrContextTag *)*ccrParam
158 
159 #define ccrBegin(x) if(!x) {x= *ccrParam=malloc(sizeof(*x)); x->ccrLine=0;}\
160  if (x) switch(x->ccrLine) { case 0:;
161 #define ccrFinish(z) } free(*ccrParam); *ccrParam=0; return (z)
162 #define ccrFinishV } free(*ccrParam); *ccrParam=0; return
163 
164 #define ccrReturn(z) \
165  do {\
166  ((struct ccrContextTag *)*ccrParam)->ccrLine=__LINE__;\
167  return (z); case __LINE__:;\
168  } while (0)
169 #define ccrReturnV \
170  do {\
171  ((struct ccrContextTag *)*ccrParam)->ccrLine=__LINE__;\
172  return; case __LINE__:;\
173  } while (0)
174 
175 #define ccrStop(z) do{ free(*ccrParam); *ccrParam=0; return (z); }while(0)
176 #define ccrStopV do{ free(*ccrParam); *ccrParam=0; return; }while(0)
177 
178 #define ccrContext void *
179 #define ccrAbort(ctx) do { free (ctx); ctx = 0; } while (0)
180 
181 #endif /* COROUTINE_H */