123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288 |
- /* classes: h_files */
- #ifndef COOP_DEFSH
- #define COOP_DEFSH
- /* Copyright (C) 1996, 1997, 1998, 1999, 2000, 2002 Free Software Foundation, Inc.
- *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License
- * along with this software; see the file COPYING. If not, write to
- * the Free Software Foundation, Inc., 59 Temple Place, Suite 330,
- * Boston, MA 02111-1307 USA
- *
- * As a special exception, the Free Software Foundation gives permission
- * for additional uses of the text contained in its release of GUILE.
- *
- * The exception is that, if you link the GUILE library with other files
- * to produce an executable, this does not by itself cause the
- * resulting executable to be covered by the GNU General Public License.
- * Your use of that executable is in no way restricted on account of
- * linking the GUILE library code into it.
- *
- * This exception does not however invalidate any other reasons why
- * the executable file might be covered by the GNU General Public License.
- *
- * This exception applies only to the code released by the
- * Free Software Foundation under the name GUILE. If you copy
- * code from other Free Software Foundation releases into a copy of
- * GUILE, as the General Public License permits, the exception does
- * not apply to the code that you add in this way. To avoid misleading
- * anyone as to the status of such modified files, you must delete
- * this exception notice from them.
- *
- * If you write modifications of your own for GUILE, it is your choice
- * whether to permit this exception to apply to your modifications.
- * If you do not wish that, delete this exception notice. */
- # ifdef TIME_WITH_SYS_TIME
- # include <sys/time.h>
- # include <time.h>
- # else
- # ifdef HAVE_SYS_TIME_H
- # include <sys/time.h>
- # else
- # ifdef HAVE_TIME_H
- # include <time.h>
- # endif
- # endif
- # endif
- #ifdef GUILE_ISELECT
- #include "libguile/iselect.h"
- #endif
- #ifdef GUILE_PTHREAD_COMPAT
- #include <pthread.h>
- #endif
- /* This file is included by threads.h, which, in turn, is included by
- libguile.h while coop-threads.h only is included by
- coop-threads.c. */
- /* The coop_t struct must be declared here, since macros in this file
- refer to the data member. */
- /* The notion of a thread is merged with the notion of a queue.
- Thread stuff: thread status (sp) and stuff to use during
- (re)initialization. Queue stuff: next thread in the queue
- (next). */
- struct qt_t;
- typedef struct coop_t {
- struct qt_t *sp; /* QuickThreads handle. */
- void *sto; /* `malloc'-allocated stack. */
- struct coop_t *next; /* Next thread in the queue. */
- struct coop_t *all_next;
- struct coop_t *all_prev;
- void *data; /* Thread local data */
- void **specific; /* Data associated with keys */
- int n_keys; /* Upper limit for keys on this thread */
-
- void *base; /* Base of stack */
- void *top; /* Top of stack */
- void *joining; /* A queue of threads waiting to join this
- thread */
- #ifdef GUILE_ISELECT
- int nfds;
- SELECT_TYPE *readfds;
- SELECT_TYPE *writefds;
- SELECT_TYPE *exceptfds;
- int timeoutp;
- struct timeval wakeup_time; /* Time to stop sleeping */
- int _errno;
- int retval;
- #else
- time_t wakeup_time; /* Time to stop sleeping */
- #endif
- #ifdef GUILE_PTHREAD_COMPAT
- pthread_t dummy_thread;
- pthread_mutex_t dummy_mutex;
- #endif
- } coop_t;
- /* A queue is a circular list of threads. The queue head is a
- designated list element. If this is a uniprocessor-only
- implementation we can store the `main' thread in this, but in a
- multiprocessor there are several `heavy' threads but only one run
- queue. A fancier implementation might have private run queues,
- which would lead to a simpler (trivial) implementation */
- typedef struct coop_q_t {
- coop_t t;
- coop_t *tail;
- } coop_q_t;
- /* A Mutex variable is made up of a owner thread, and a queue of threads
- waiting on the mutex */
- typedef struct coop_m {
- coop_t *owner; /* Mutex owner */
- coop_q_t waiting; /* Queue of waiting threads */
- } coop_m;
- typedef int coop_mattr;
- typedef coop_m scm_mutex_t;
- extern int coop_mutex_init (coop_m*);
- extern int coop_new_mutex_init (coop_m*, coop_mattr*);
- extern int coop_mutex_lock (coop_m*);
- extern int coop_mutex_trylock (coop_m*);
- extern int coop_mutex_unlock (coop_m*);
- extern int coop_mutex_destroy (coop_m*);
- #define scm_mutex_init coop_mutex_init
- #define scm_mutex_lock coop_mutex_lock
- #define scm_mutex_trylock coop_mutex_lock
- #define scm_mutex_unlock coop_mutex_unlock
- #define scm_mutex_destroy coop_mutex_destroy
- /* A Condition variable is made up of a list of threads waiting on the
- condition. */
- typedef struct coop_c {
- coop_q_t waiting; /* Queue of waiting threads */
- } coop_c;
- typedef int coop_cattr;
- typedef coop_c scm_cond_t;
- #ifndef HAVE_STRUCT_TIMESPEC
- /* POSIX.4 structure for a time value. This is like a `struct timeval' but
- has nanoseconds instead of microseconds. */
- struct timespec
- {
- long int tv_sec; /* Seconds. */
- long int tv_nsec; /* Nanoseconds. */
- };
- #endif
- extern int coop_condition_variable_init (coop_c*);
- extern int coop_new_condition_variable_init (coop_c*, coop_cattr*);
- extern int coop_condition_variable_wait_mutex (coop_c*, coop_m*);
- extern int coop_condition_variable_timed_wait_mutex (coop_c*,
- coop_m*,
- const struct timespec *abstime);
- extern int coop_condition_variable_signal (coop_c*);
- extern int coop_condition_variable_destroy (coop_c*);
- #define scm_cond_init coop_new_condition_variable_init
- #define scm_cond_wait coop_condition_variable_wait_mutex
- #define scm_cond_timedwait coop_condition_variable_timed_wait_mutex
- #define scm_cond_signal coop_condition_variable_signal
- #define scm_cond_broadcast coop_condition_variable_signal /* yes */
- #define scm_cond_destroy coop_condition_variable_destroy
- typedef int coop_k;
- typedef coop_k scm_key_t;
- extern int coop_key_create (coop_k *keyp, void (*destructor) (void *value));
- extern int coop_setspecific (coop_k key, const void *value);
- extern void *coop_getspecific (coop_k key);
- extern int coop_key_delete (coop_k);
- #define scm_key_create coop_key_create
- #define scm_setspecific coop_setspecific
- #define scm_getspecific coop_getspecific
- #define scm_key_delete coop_key_delete
- extern coop_t *coop_global_curr; /* Currently-executing thread. */
- extern void coop_join (coop_t *t);
- extern void coop_yield (void);
- extern size_t scm_switch_counter;
- extern size_t scm_thread_count;
- /* Some iselect functions. */
- /* I'm not sure whether these three declarations should be here.
- They're really defined in iselect.c, so you'd think they'd go in
- iselect.h, but they use coop_t, defined above, which uses things
- defined in iselect.h. Basically, we're making at best a flailing
- (and failing) attempt at modularity here, and I don't have time to
- rethink this at the moment. This code awaits a Hero. --JimB */
- #ifdef GUILE_ISELECT
- void coop_timeout_qinsert (coop_q_t *, coop_t *);
- #endif
- extern coop_t *coop_next_runnable_thread (void);
- extern coop_t *coop_wait_for_runnable_thread_now (struct timeval *);
- extern coop_t *coop_wait_for_runnable_thread (void);
- /* Cooperative threads don't need to have these defined */
- #define SCM_THREAD_CRITICAL_SECTION_START
- #define SCM_THREAD_CRITICAL_SECTION_END
- #define SCM_NO_CRITICAL_SECTION_OWNER 0
- #define SCM_THREAD_SWITCH_COUNT 50 /* was 10 /mdj */
- #define SCM_THREAD_DEFER
- #define SCM_THREAD_ALLOW
- #define SCM_THREAD_REDEFER
- #define SCM_THREAD_REALLOW_1
- #define SCM_THREAD_REALLOW_2
- #if 0
- #define SCM_THREAD_SWITCHING_CODE \
- do { \
- if (scm_thread_count > 1) \
- coop_yield(); \
- } while (0)
- #else
- #define SCM_THREAD_SWITCHING_CODE \
- do { \
- if (scm_thread_count > 1) \
- { \
- scm_switch_counter--; \
- if (scm_switch_counter == 0) \
- { \
- scm_switch_counter = SCM_THREAD_SWITCH_COUNT; \
- coop_yield(); \
- } \
- } \
- } while (0)
- #endif
- /* For pthreads, this is a value associated with a specific key.
- * For coop, we use a special field for increased efficiency.
- */
- #define SCM_THREAD_LOCAL_DATA (coop_global_curr->data)
- #define SCM_SET_THREAD_LOCAL_DATA(ptr) (coop_global_curr->data = (ptr))
- #endif /* COOP_DEFSH */
- /*
- Local Variables:
- c-file-style: "gnu"
- End:
- */
|