summaryrefslogtreecommitdiff
path: root/libguile/null-threads.c
diff options
context:
space:
mode:
authorMarius Vollmer <mvo@zagadka.de>2005-03-02 20:42:01 +0000
committerMarius Vollmer <mvo@zagadka.de>2005-03-02 20:42:01 +0000
commit9de87eea47536e25ef99bc25f07afdd759ee3575 (patch)
tree936d52588ace67469aba68f6ad47cbf1a12462ca /libguile/null-threads.c
parentcb1cfc42a4f5ac4d60a64b425795432a0388ad7e (diff)
downloadguile-9de87eea47536e25ef99bc25f07afdd759ee3575.tar.gz
See ChangeLog from 2005-03-02.
Diffstat (limited to 'libguile/null-threads.c')
-rw-r--r--libguile/null-threads.c327
1 files changed, 30 insertions, 297 deletions
diff --git a/libguile/null-threads.c b/libguile/null-threads.c
index 7f2eb8b43..7b1f85fc9 100644
--- a/libguile/null-threads.c
+++ b/libguile/null-threads.c
@@ -15,318 +15,51 @@
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
+#include <stdlib.h>
+#include "libguile/_scm.h"
-
+#if SCM_USE_NULL_THREADS
-#include "libguile/validate.h"
-#include "libguile/root.h"
-#include "libguile/stackchk.h"
-#include "libguile/async.h"
-#include <sys/time.h>
-#include <sys/types.h>
-#include <time.h>
-#include <stdio.h>
+#include "libguile/null-threads.h"
-void *scm_null_threads_data;
+static scm_i_pthread_key_t *all_keys = NULL;
-static SCM main_thread;
-
-typedef struct {
- int level;
-} scm_null_mutex;
-
-typedef struct {
- int signalled;
-} scm_null_cond;
-
-void
-scm_threads_init (SCM_STACKITEM *i)
-{
- scm_tc16_thread = scm_make_smob_type ("thread", 0);
- scm_tc16_mutex = scm_make_smob_type ("mutex", sizeof (scm_null_mutex));
- scm_tc16_condvar = scm_make_smob_type ("condition-variable",
- sizeof (scm_null_cond));
-
- main_thread = scm_permanent_object (scm_cell (scm_tc16_thread, 0));
- scm_null_threads_data = NULL;
-}
-
-#ifdef __ia64__
-# define SCM_MARK_BACKING_STORE() do { \
- ucontext_t ctx; \
- SCM_STACKITEM * top, * bot; \
- getcontext (&ctx); \
- scm_mark_locations ((SCM_STACKITEM *) &ctx.uc_mcontext, \
- ((size_t) (sizeof (SCM_STACKITEM) - 1 + sizeof ctx.uc_mcontext) \
- / sizeof (SCM_STACKITEM))); \
- bot = (SCM_STACKITEM *) __libc_ia64_register_backing_store_base; \
- top = (SCM_STACKITEM *) ctx.uc_mcontext.sc_ar_bsp; \
- scm_mark_locations (bot, top - bot); } while (0)
-#else
-# define SCM_MARK_BACKING_STORE()
-#endif
-
-void
-scm_threads_mark_stacks (void)
-{
- /* Mark objects on the C stack. */
- SCM_FLUSH_REGISTER_WINDOWS;
- /* This assumes that all registers are saved into the jmp_buf */
- setjmp (scm_save_regs_gc_mark);
- scm_mark_locations ((SCM_STACKITEM *) scm_save_regs_gc_mark,
- ( (size_t) (sizeof (SCM_STACKITEM) - 1 +
- sizeof scm_save_regs_gc_mark)
- / sizeof (SCM_STACKITEM)));
-
- {
- unsigned long stack_len = scm_stack_size (scm_stack_base);
-#if SCM_STACK_GROWS_UP
- scm_mark_locations (scm_stack_base, stack_len);
-#else
- scm_mark_locations (scm_stack_base - stack_len, stack_len);
-#endif
- }
- SCM_MARK_BACKING_STORE();
-}
-
-SCM
-scm_call_with_new_thread (SCM argl)
-#define FUNC_NAME s_call_with_new_thread
-{
- SCM_MISC_ERROR ("threads are not supported in this version of Guile",
- SCM_EOL);
- return SCM_BOOL_F;
-}
-#undef FUNC_NAME
-
-SCM
-scm_spawn_thread (scm_t_catch_body body, void *body_data,
- scm_t_catch_handler handler, void *handler_data)
-{
- scm_misc_error ("scm_spawn_thread",
- "threads are not supported in this version of Guile",
- SCM_EOL);
- return SCM_BOOL_F;
-}
-
-SCM
-scm_current_thread (void)
-{
- return main_thread;
-}
-
-SCM
-scm_all_threads (void)
-{
- return scm_list_1 (main_thread);
-}
-
-scm_root_state *
-scm_i_thread_root (SCM thread)
-{
- return (scm_root_state *)scm_null_threads_data;
-}
-
-SCM
-scm_join_thread (SCM thread)
-#define FUNC_NAME s_join_thread
-{
- SCM_MISC_ERROR ("threads are not supported in this version of Guile",
- SCM_EOL);
- return SCM_BOOL_F;
-}
-#undef FUNC_NAME
-
-int
-scm_c_thread_exited_p (SCM thread)
-#define FUNC_NAME s_scm_thread_exited_p
-{
- return 0;
-}
-#undef FUNC_NAME
-
-SCM
-scm_yield (void)
-{
- return SCM_BOOL_T;
-}
-
-SCM
-scm_make_mutex (void)
-{
- SCM m = scm_make_smob (scm_tc16_mutex);
- scm_null_mutex *mx = SCM_MUTEX_DATA(m);
- mx->level = 0;
- return m;
-}
-
-SCM
-scm_lock_mutex (SCM m)
-{
- scm_null_mutex *mx;
- SCM_ASSERT (SCM_MUTEXP (m), m, SCM_ARG1, s_lock_mutex);
- mx = SCM_MUTEX_DATA(m);
- mx->level++;
- return SCM_BOOL_T;
-}
-
-SCM
-scm_try_mutex (SCM m)
-{
- return scm_lock_mutex (m); /* will always succeed right away. */
-}
-
-SCM
-scm_unlock_mutex (SCM m)
-{
- scm_null_mutex *mx;
- SCM_ASSERT (SCM_MUTEXP (m), m, SCM_ARG1, s_unlock_mutex);
- mx = SCM_MUTEX_DATA(m);
- if (mx->level == 0)
- scm_misc_error (s_unlock_mutex, "mutex is not locked", SCM_EOL);
- mx->level--;
- return SCM_BOOL_T;
-}
-
-SCM
-scm_make_condition_variable (void)
-{
- scm_null_cond *cv;
- SCM c = scm_make_smob (scm_tc16_condvar);
- cv = SCM_CONDVAR_DATA (c);
- cv->signalled = 0;
- return c;
-}
-
-/* Subtract the `struct timeval' values X and Y,
- storing the result in RESULT. Might modify Y.
- Return 1 if the difference is negative or zero, otherwise 0. */
-
-static int
-timeval_subtract (result, x, y)
- struct timeval *result, *x, *y;
+static void
+destroy_keys (void)
{
- /* Perform the carry for the later subtraction by updating Y. */
- if (x->tv_usec < y->tv_usec) {
- int nsec = (y->tv_usec - x->tv_usec) / 1000000 + 1;
- y->tv_usec -= 1000000 * nsec;
- y->tv_sec += nsec;
- }
- if (x->tv_usec - y->tv_usec > 1000000) {
- int nsec = (x->tv_usec - y->tv_usec) / 1000000;
- y->tv_usec += 1000000 * nsec;
- y->tv_sec -= nsec;
- }
-
- /* Compute the time remaining to wait.
- `tv_usec' is certainly positive. */
- result->tv_sec = x->tv_sec - y->tv_sec;
- result->tv_usec = x->tv_usec - y->tv_usec;
-
- /* Return 1 if result is negative or zero. */
- return x->tv_sec < y->tv_sec
- || (result->tv_sec == 0 && result->tv_usec == 0);
-}
-
-SCM
-scm_timed_wait_condition_variable (SCM c, SCM m, SCM t)
-#define FUNC_NAME s_wait_condition_variable
-{
- scm_null_cond *cv;
- struct timeval waittime;
-
- SCM_ASSERT (SCM_CONDVARP (c),
- c,
- SCM_ARG1,
- s_wait_condition_variable);
- SCM_ASSERT (SCM_MUTEXP (m),
- m,
- SCM_ARG2,
- s_wait_condition_variable);
- if (!SCM_UNBNDP (t))
- {
- if (scm_is_pair (t))
- {
- SCM_VALIDATE_UINT_COPY (3, SCM_CAR(t), waittime.tv_sec);
- SCM_VALIDATE_UINT_COPY (3, SCM_CDR(t), waittime.tv_usec);
- }
- else
- {
- SCM_VALIDATE_UINT_COPY (3, t, waittime.tv_sec);
- waittime.tv_usec = 0;
- }
- }
-
- cv = SCM_CONDVAR_DATA (c);
+ scm_i_pthread_key_t *key;
+ int again;
- scm_unlock_mutex (m);
- while (!cv->signalled)
- {
- if (SCM_UNBNDP (t))
- select (0, NULL, NULL, NULL, NULL);
- else
+ do {
+ again = 0;
+ for (key = all_keys; key; key = key->next)
+ if (key->value && key->destr_func)
{
- struct timeval now, then, diff;
- then = waittime;
- gettimeofday (&now, NULL);
- if (timeval_subtract (&diff, &then, &now))
- break;
- select (0, NULL, NULL, NULL, &diff);
+ void *v = key->value;
+ key->value = NULL;
+ key->destr_func (v);
+ again = 1;
}
- SCM_ASYNC_TICK;
- }
- scm_lock_mutex (m);
- if (cv->signalled)
- {
- cv->signalled = 0;
- return SCM_BOOL_T;
- }
- return SCM_BOOL_F;
+ } while (again);
}
-#undef FUNC_NAME
-SCM
-scm_signal_condition_variable (SCM c)
+int
+scm_i_pthread_key_create (scm_i_pthread_key_t *key,
+ void (*destr_func) (void *))
{
- scm_null_cond *cv;
- SCM_ASSERT (SCM_CONDVARP (c),
- c,
- SCM_ARG1,
- s_signal_condition_variable);
- cv = SCM_CONDVAR_DATA (c);
- cv->signalled = 1;
- return SCM_BOOL_T;
-}
+ if (all_keys == NULL)
+ atexit (destroy_keys);
-SCM
-scm_broadcast_condition_variable (SCM c)
-{
- return scm_signal_condition_variable (c); /* only one thread anyway. */
-}
+ key->next = all_keys;
+ all_keys = key;
+ key->value = NULL;
+ key->destr_func = destr_func;
-unsigned long
-scm_thread_usleep (unsigned long usec)
-{
- struct timeval timeout;
- timeout.tv_sec = 0;
- timeout.tv_usec = usec;
- select (0, NULL, NULL, NULL, &timeout);
- return 0; /* Maybe we should calculate actual time slept,
- but this is faster... :) */
+ return 0;
}
-unsigned long
-scm_thread_sleep (unsigned long sec)
-{
- time_t now = time (NULL);
- struct timeval timeout;
- unsigned long slept;
- timeout.tv_sec = sec;
- timeout.tv_usec = 0;
- select (0, NULL, NULL, NULL, &timeout);
- slept = time (NULL) - now;
- return slept > sec ? 0 : sec - slept;
-}
+#endif /* SCM_USE_NULL_THREADS */
+
/*
Local Variables: