*** empty log message ***
[bpt/guile.git] / libguile / null-threads.c
CommitLineData
3d527b27
MV
1/* Copyright (C) 2002 Free Software Foundation, Inc.
2 *
3 * This program is free software; you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License as published by
5 * the Free Software Foundation; either version 2, or (at your option)
6 * any later version.
7 *
8 * This program is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 * GNU General Public License for more details.
12 *
13 * You should have received a copy of the GNU General Public License
14 * along with this software; see the file COPYING. If not, write to
15 * the Free Software Foundation, Inc., 59 Temple Place, Suite 330,
16 * Boston, MA 02111-1307 USA
17 *
18 * As a special exception, the Free Software Foundation gives permission
19 * for additional uses of the text contained in its release of GUILE.
20 *
21 * The exception is that, if you link the GUILE library with other files
22 * to produce an executable, this does not by itself cause the
23 * resulting executable to be covered by the GNU General Public License.
24 * Your use of that executable is in no way restricted on account of
25 * linking the GUILE library code into it.
26 *
27 * This exception does not however invalidate any other reasons why
28 * the executable file might be covered by the GNU General Public License.
29 *
30 * This exception applies only to the code released by the
31 * Free Software Foundation under the name GUILE. If you copy
32 * code from other Free Software Foundation releases into a copy of
33 * GUILE, as the General Public License permits, the exception does
34 * not apply to the code that you add in this way. To avoid misleading
35 * anyone as to the status of such modified files, you must delete
36 * this exception notice from them.
37 *
38 * If you write modifications of your own for GUILE, it is your choice
39 * whether to permit this exception to apply to your modifications.
40 * If you do not wish that, delete this exception notice. */
41
42
43\f
44
45#include "libguile/validate.h"
46#include "libguile/root.h"
47#include "libguile/stackchk.h"
48#include "libguile/async.h"
64e00566
MV
49#include <sys/time.h>
50#include <sys/types.h>
51#include <time.h>
4b9154e7 52#include <stdio.h>
3d527b27
MV
53
54void *scm_null_threads_data;
55
56static SCM main_thread;
57
4b9154e7
MV
58typedef struct {
59 int level;
60} scm_null_mutex;
61
62typedef struct {
63 int signalled;
64} scm_null_cond;
65
3d527b27
MV
66void
67scm_threads_init (SCM_STACKITEM *i)
68{
4b9154e7
MV
69 scm_tc16_thread = scm_make_smob_type ("thread", 0);
70 scm_tc16_mutex = scm_make_smob_type ("mutex", sizeof (scm_null_mutex));
71 scm_tc16_condvar = scm_make_smob_type ("condition-variable",
72 sizeof (scm_null_cond));
73
3d527b27
MV
74 main_thread = scm_permanent_object (scm_cell (scm_tc16_thread, 0));
75 scm_null_threads_data = NULL;
76}
77
78#ifdef __ia64__
79# define SCM_MARK_BACKING_STORE() do { \
80 ucontext_t ctx; \
81 SCM_STACKITEM * top, * bot; \
82 getcontext (&ctx); \
83 scm_mark_locations ((SCM_STACKITEM *) &ctx.uc_mcontext, \
84 ((size_t) (sizeof (SCM_STACKITEM) - 1 + sizeof ctx.uc_mcontext) \
85 / sizeof (SCM_STACKITEM))); \
86 bot = (SCM_STACKITEM *) __libc_ia64_register_backing_store_base; \
87 top = (SCM_STACKITEM *) ctx.uc_mcontext.sc_ar_bsp; \
88 scm_mark_locations (bot, top - bot); } while (0)
89#else
90# define SCM_MARK_BACKING_STORE()
91#endif
92
93void
94scm_threads_mark_stacks (void)
95{
96 /* Mark objects on the C stack. */
97 SCM_FLUSH_REGISTER_WINDOWS;
98 /* This assumes that all registers are saved into the jmp_buf */
99 setjmp (scm_save_regs_gc_mark);
100 scm_mark_locations ((SCM_STACKITEM *) scm_save_regs_gc_mark,
101 ( (size_t) (sizeof (SCM_STACKITEM) - 1 +
102 sizeof scm_save_regs_gc_mark)
103 / sizeof (SCM_STACKITEM)));
104
105 {
106 unsigned long stack_len = scm_stack_size (scm_stack_base);
b075613e 107#if SCM_STACK_GROWS_UP
3d527b27
MV
108 scm_mark_locations (scm_stack_base, stack_len);
109#else
110 scm_mark_locations (scm_stack_base - stack_len, stack_len);
111#endif
112 }
113 SCM_MARK_BACKING_STORE();
114}
115
116SCM
117scm_call_with_new_thread (SCM argl)
118#define FUNC_NAME s_call_with_new_thread
119{
120 SCM_MISC_ERROR ("threads are not supported in this version of Guile",
121 SCM_EOL);
122 return SCM_BOOL_F;
123}
124#undef FUNC_NAME
125
126SCM
127scm_spawn_thread (scm_t_catch_body body, void *body_data,
128 scm_t_catch_handler handler, void *handler_data)
129{
130 scm_misc_error ("scm_spawn_thread",
131 "threads are not supported in this version of Guile",
132 SCM_EOL);
133 return SCM_BOOL_F;
134}
135
136SCM
137scm_current_thread (void)
138{
139 return main_thread;
140}
141
142SCM
143scm_all_threads (void)
144{
145 return scm_list_1 (main_thread);
146}
147
148scm_root_state *
149scm_i_thread_root (SCM thread)
150{
151 return (scm_root_state *)scm_null_threads_data;
152}
153
154SCM
155scm_join_thread (SCM thread)
156#define FUNC_NAME s_join_thread
157{
158 SCM_MISC_ERROR ("threads are not supported in this version of Guile",
159 SCM_EOL);
160 return SCM_BOOL_F;
161}
162#undef FUNC_NAME
163
3d527b27 164int
4b9154e7
MV
165scm_c_thread_exited_p (SCM thread)
166#define FUNC_NAME s_scm_thread_exited_p
3d527b27 167{
3d527b27
MV
168 return 0;
169}
4b9154e7 170#undef FUNC_NAME
3d527b27 171
4b9154e7
MV
172SCM
173scm_yield (void)
3d527b27 174{
4b9154e7 175 return SCM_BOOL_T;
3d527b27
MV
176}
177
178SCM
179scm_make_mutex (void)
180{
181 SCM m = scm_make_smob (scm_tc16_mutex);
4b9154e7
MV
182 scm_null_mutex *mx = SCM_MUTEX_DATA(m);
183 mx->level = 0;
3d527b27
MV
184 return m;
185}
186
187SCM
188scm_lock_mutex (SCM m)
189{
4b9154e7 190 scm_null_mutex *mx;
3d527b27 191 SCM_ASSERT (SCM_MUTEXP (m), m, SCM_ARG1, s_lock_mutex);
4b9154e7
MV
192 mx = SCM_MUTEX_DATA(m);
193 mx->level++;
3d527b27
MV
194 return SCM_BOOL_T;
195}
196
4b9154e7
MV
197SCM
198scm_try_mutex (SCM m)
199{
200 return scm_lock_mutex (m); /* will always succeed right away. */
201}
202
3d527b27
MV
203SCM
204scm_unlock_mutex (SCM m)
205{
4b9154e7 206 scm_null_mutex *mx;
3d527b27 207 SCM_ASSERT (SCM_MUTEXP (m), m, SCM_ARG1, s_unlock_mutex);
4b9154e7
MV
208 mx = SCM_MUTEX_DATA(m);
209 if (mx->level == 0)
3d527b27 210 scm_misc_error (s_unlock_mutex, "mutex is not locked", SCM_EOL);
4b9154e7 211 mx->level--;
3d527b27
MV
212 return SCM_BOOL_T;
213}
214
4b9154e7
MV
215SCM
216scm_make_condition_variable (void)
3d527b27 217{
4b9154e7
MV
218 scm_null_cond *cv;
219 SCM c = scm_make_smob (scm_tc16_condvar);
220 cv = SCM_CONDVAR_DATA (c);
221 cv->signalled = 0;
222 return c;
3d527b27
MV
223}
224
4b9154e7
MV
225/* Subtract the `struct timeval' values X and Y,
226 storing the result in RESULT. Might modify Y.
227 Return 1 if the difference is negative or zero, otherwise 0. */
3d527b27 228
4b9154e7
MV
229static int
230timeval_subtract (result, x, y)
231 struct timeval *result, *x, *y;
3d527b27 232{
4b9154e7
MV
233 /* Perform the carry for the later subtraction by updating Y. */
234 if (x->tv_usec < y->tv_usec) {
235 int nsec = (y->tv_usec - x->tv_usec) / 1000000 + 1;
236 y->tv_usec -= 1000000 * nsec;
237 y->tv_sec += nsec;
238 }
239 if (x->tv_usec - y->tv_usec > 1000000) {
240 int nsec = (x->tv_usec - y->tv_usec) / 1000000;
241 y->tv_usec += 1000000 * nsec;
242 y->tv_sec -= nsec;
243 }
244
245 /* Compute the time remaining to wait.
246 `tv_usec' is certainly positive. */
247 result->tv_sec = x->tv_sec - y->tv_sec;
248 result->tv_usec = x->tv_usec - y->tv_usec;
249
250 /* Return 1 if result is negative or zero. */
251 return x->tv_sec < y->tv_sec
252 || (result->tv_sec == 0 && result->tv_usec == 0);
3d527b27
MV
253}
254
255SCM
4b9154e7
MV
256scm_timed_wait_condition_variable (SCM c, SCM m, SCM t)
257#define FUNC_NAME s_wait_condition_variable
3d527b27 258{
4b9154e7
MV
259 scm_null_cond *cv;
260 struct timeval waittime;
3d527b27 261
3d527b27
MV
262 SCM_ASSERT (SCM_CONDVARP (c),
263 c,
264 SCM_ARG1,
265 s_wait_condition_variable);
266 SCM_ASSERT (SCM_MUTEXP (m),
267 m,
268 SCM_ARG2,
269 s_wait_condition_variable);
4b9154e7
MV
270 if (!SCM_UNBNDP (t))
271 {
272 if (SCM_CONSP (t))
273 {
274 SCM_VALIDATE_UINT_COPY (3, SCM_CAR(t), waittime.tv_sec);
275 SCM_VALIDATE_UINT_COPY (3, SCM_CDR(t), waittime.tv_usec);
276 }
277 else
278 {
279 SCM_VALIDATE_UINT_COPY (3, t, waittime.tv_sec);
280 waittime.tv_usec = 0;
281 }
282 }
283
284 cv = SCM_CONDVAR_DATA (c);
285
286 scm_unlock_mutex (m);
287 while (!cv->signalled)
288 {
289 if (SCM_UNBNDP (t))
290 select (0, NULL, NULL, NULL, NULL);
291 else
292 {
293 struct timeval now, then, diff;
294 then = waittime;
295 gettimeofday (&now, NULL);
296 if (timeval_subtract (&diff, &then, &now))
297 break;
298 select (0, NULL, NULL, NULL, &diff);
299 }
300 SCM_ASYNC_TICK;
301 }
302 scm_lock_mutex (m);
303 if (cv->signalled)
304 {
305 cv->signalled = 0;
306 return SCM_BOOL_T;
307 }
308 return SCM_BOOL_F;
3d527b27 309}
4b9154e7 310#undef FUNC_NAME
3d527b27
MV
311
312SCM
313scm_signal_condition_variable (SCM c)
314{
4b9154e7 315 scm_null_cond *cv;
3d527b27
MV
316 SCM_ASSERT (SCM_CONDVARP (c),
317 c,
318 SCM_ARG1,
319 s_signal_condition_variable);
4b9154e7
MV
320 cv = SCM_CONDVAR_DATA (c);
321 cv->signalled = 1;
3d527b27
MV
322 return SCM_BOOL_T;
323}
324
4b9154e7
MV
325SCM
326scm_broadcast_condition_variable (SCM c)
327{
328 return scm_signal_condition_variable (c); /* only one thread anyway. */
329}
330
3d527b27
MV
331unsigned long
332scm_thread_usleep (unsigned long usec)
333{
334 struct timeval timeout;
335 timeout.tv_sec = 0;
336 timeout.tv_usec = usec;
337 select (0, NULL, NULL, NULL, &timeout);
338 return 0; /* Maybe we should calculate actual time slept,
339 but this is faster... :) */
340}
341
342unsigned long
343scm_thread_sleep (unsigned long sec)
344{
345 time_t now = time (NULL);
346 struct timeval timeout;
347 unsigned long slept;
348 timeout.tv_sec = sec;
349 timeout.tv_usec = 0;
350 select (0, NULL, NULL, NULL, &timeout);
351 slept = time (NULL) - now;
352 return slept > sec ? 0 : sec - slept;
353}
354
355/*
356 Local Variables:
357 c-file-style: "gnu"
358 End:
359*/