pthread.c
上传用户:tsgydb
上传日期:2007-04-14
资源大小:10674k
文件大小:8k
源码类别:

MySQL数据库

开发平台:

Visual C++

  1. /* ==== pthread.c ============================================================
  2.  * Copyright (c) 1993, 1994 by Chris Provenzano, proven@mit.edu
  3.  * All rights reserved.
  4.  *
  5.  * Redistribution and use in source and binary forms, with or without
  6.  * modification, are permitted provided that the following conditions
  7.  * are met:
  8.  * 1. Redistributions of source code must retain the above copyright
  9.  *    notice, this list of conditions and the following disclaimer.
  10.  * 2. Redistributions in binary form must reproduce the above copyright
  11.  *    notice, this list of conditions and the following disclaimer in the
  12.  *    documentation and/or other materials provided with the distribution.
  13.  * 3. All advertising materials mentioning features or use of this software
  14.  *    must display the following acknowledgement:
  15.  *  This product includes software developed by Chris Provenzano.
  16.  * 4. The name of Chris Provenzano may not be used to endorse or promote 
  17.  *   products derived from this software without specific prior written
  18.  *   permission.
  19.  *
  20.  * THIS SOFTWARE IS PROVIDED BY CHRIS PROVENZANO ``AS IS'' AND
  21.  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  22.  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  23.  * ARE DISCLAIMED.  IN NO EVENT SHALL CHRIS PROVENZANO BE LIABLE FOR ANY 
  24.  * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
  25.  * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR 
  26.  * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
  27.  * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 
  28.  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
  29.  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 
  30.  * SUCH DAMAGE.
  31.  *
  32.  * Description : Pthread functions.
  33.  *
  34.  *  1.00 93/07/26 proven
  35.  *      -Started coding this file.
  36.  */
  37. #ifndef lint
  38. static const char rcsid[] = "$Id$";
  39. #endif
  40. #include <pthread.h>
  41. #include <stdlib.h>
  42. #include <signal.h>
  43. #include <errno.h>
  44. #include <string.h>
  45. #include <sched.h>
  46. /* ==========================================================================
  47.  * sched_yield()
  48.  */
  49. int sched_yield()
  50. {
  51. sig_handler_fake(SIGVTALRM);
  52. return(OK);
  53. }
  54. /* ==========================================================================
  55.  * pthread_yield()
  56.  */
  57. void pthread_yield()
  58. {
  59. sig_handler_fake(SIGVTALRM);
  60. }
  61. /* ==========================================================================
  62.  * pthread_self()
  63.  */
  64. pthread_t pthread_self()
  65. {
  66. return(pthread_run);
  67. }
  68. /* ==========================================================================
  69.  * pthread_equal()
  70.  */
  71. int pthread_equal(pthread_t t1, pthread_t t2)
  72. {
  73. return(t1 == t2);
  74. }
  75. /* ==========================================================================
  76.  * pthread_exit()
  77.  */
  78. extern void pthread_cleanupspecific(void);
  79. void pthread_exit(void *status)
  80. {
  81. pthread_t pthread;
  82. /* Save return value */
  83. pthread_run->ret = status;
  84. /* First execute all cleanup handlers */
  85. while (pthread_run->cleanup) {
  86. pthread_cleanup_pop(1);
  87. }
  88. /* Don't forget the cleanup attr */
  89. if (pthread_run->attr.cleanup_attr) {
  90. pthread_run->attr.cleanup_attr(pthread_run->attr.arg_attr);
  91. }
  92. /* Next run thread-specific data desctructors */
  93. if (pthread_run->specific_data) {
  94. pthread_cleanupspecific();
  95. }
  96. pthread_sched_prevent();
  97. if (!(pthread_run->attr.flags & PTHREAD_DETACHED)) {
  98. /*
  99.  * Are there any threads joined to this one,
  100.  * if so wake them and let them detach this thread.
  101.  */
  102. while (pthread = pthread_queue_deq(&(pthread_run->join_queue))) {
  103. pthread_prio_queue_enq(pthread_current_prio_queue, pthread);
  104. pthread->state = PS_RUNNING;
  105. }
  106. pthread_queue_enq(&pthread_dead_queue, pthread_run);
  107. pthread_resched_resume(PS_DEAD);
  108. } else {
  109. pthread_queue_enq(&pthread_alloc_queue, pthread_run);
  110. pthread_resched_resume(PS_UNALLOCED);
  111. }
  112. /* This thread will never run again */
  113. PANIC();
  114. }
  115. /*----------------------------------------------------------------------
  116.  * Function: __pthread_is_valid
  117.  * Purpose:  Scan the list of threads to see if a specified thread exists
  118.  * Args:
  119.  *     pthread = The thread to scan for
  120.  * Returns:
  121.  *     int     = 1 if found, 0 if not
  122.  * Notes:
  123.  *     The kernel is assumed to be locked
  124.  *----------------------------------------------------------------------*/
  125. int
  126. __pthread_is_valid( pthread_t pthread )
  127. {
  128. int rtn = 0; /* Assume not found */
  129. pthread_t t;
  130. for( t = pthread_link_list; t; t = t->pll ) {
  131. if( t == pthread ) {
  132. rtn = 1; /* Found it */
  133. break;
  134. }
  135. }
  136. return rtn;
  137. }
  138. /* ==========================================================================
  139.  * __pthread_free()
  140.  */
  141. static inline void __pthread_free(pthread_t new_thread)
  142. {
  143. pthread_sched_prevent();
  144. new_thread->state = PS_UNALLOCED;
  145. new_thread->attr.stacksize_attr = 0; 
  146. new_thread->attr.stackaddr_attr = NULL; 
  147. pthread_queue_enq(&pthread_alloc_queue, new_thread);
  148. pthread_sched_resume();
  149. }
  150. /* ==========================================================================
  151.  * __pthread_alloc()
  152.  */
  153. /* static inline pthread_t __pthread_alloc(const pthread_attr_t *attr) */
  154. static pthread_t __pthread_alloc(const pthread_attr_t *attr)
  155. {
  156. pthread_t thread;
  157. void * stack;
  158. void * old;
  159. pthread_sched_prevent();
  160. thread = pthread_queue_deq(&pthread_alloc_queue);
  161. pthread_sched_resume();
  162. if (thread) {
  163. if (stack = attr->stackaddr_attr) {
  164. __machdep_stack_repl(&(thread->machdep_data), stack);
  165. } else {
  166. if ((__machdep_stack_get(&(thread->machdep_data)) == NULL)
  167.   || (attr->stacksize_attr > thread->attr.stacksize_attr)) {
  168. if (stack = __machdep_stack_alloc(attr->stacksize_attr)) {
  169. __machdep_stack_repl(&(thread->machdep_data), stack);
  170. } else {
  171. __pthread_free(thread);
  172. thread = NULL;
  173. }
  174. }
  175. }
  176. } else {
  177. /* We should probable allocate several for efficiency */
  178. if (thread = (pthread_t)malloc(sizeof(struct pthread))) {
  179. /* Link new thread into list of all threads */
  180. pthread_sched_prevent();
  181. thread->state = PS_UNALLOCED;
  182. thread->pll = pthread_link_list;
  183. pthread_link_list = thread;
  184. pthread_sched_resume();
  185. if ((stack = attr->stackaddr_attr) ||
  186.   (stack = __machdep_stack_alloc(attr->stacksize_attr))) {
  187. __machdep_stack_set(&(thread->machdep_data), stack);
  188. } else {
  189. __machdep_stack_set(&(thread->machdep_data), NULL);
  190. __pthread_free(thread);
  191. thread = NULL;
  192. }
  193. }
  194. }
  195. return(thread);
  196. }
  197. /* ==========================================================================
  198.  * pthread_create()
  199.  *
  200.  * After the new thread structure is allocated and set up, it is added to
  201.  * pthread_run_next_queue, which requires a sig_prevent(),
  202.  * sig_check_and_resume()
  203.  */
  204. int pthread_create(pthread_t *thread, const pthread_attr_t *attr,
  205.   void * (*start_routine)(void *), void *arg)
  206. {
  207. pthread_t new_thread;
  208. int nsec = 100000000;
  209. int retval = OK;
  210. if (! attr) 
  211. attr = &pthread_attr_default; 
  212. if (new_thread = __pthread_alloc(attr)) {
  213. __machdep_pthread_create(&(new_thread->machdep_data),
  214.   start_routine, arg, attr->stacksize_attr, nsec, 0);
  215. memcpy(&new_thread->attr, attr, sizeof(pthread_attr_t));
  216. if (new_thread->attr.flags & PTHREAD_INHERIT_SCHED) {
  217. new_thread->pthread_priority = pthread_run->pthread_priority; 
  218. new_thread->attr.sched_priority = pthread_run->pthread_priority;
  219. new_thread->attr.schedparam_policy = 
  220.   pthread_run->attr.schedparam_policy;
  221. }  else {
  222. new_thread->pthread_priority = new_thread->attr.sched_priority;
  223. }
  224. if (!(new_thread->attr.flags & PTHREAD_NOFLOAT)) {
  225. machdep_save_float_state(new_thread);
  226. }
  227. /* Initialize signalmask */
  228. new_thread->sigmask = pthread_run->sigmask;
  229. sigemptyset(&(new_thread->sigpending));
  230. new_thread->sigcount = 0;
  231. pthread_queue_init(&(new_thread->join_queue));
  232. new_thread->specific_data = NULL;
  233. new_thread->specific_data_count = 0;
  234. new_thread->cleanup = NULL;
  235. new_thread->queue = NULL;
  236. new_thread->next = NULL;
  237. new_thread->flags = 0;
  238. /* PTHREADS spec says we start with cancellability on and deferred */
  239. SET_PF_CANCEL_STATE(new_thread, PTHREAD_CANCEL_ENABLE);
  240. SET_PF_CANCEL_TYPE(new_thread, PTHREAD_CANCEL_DEFERRED);
  241. new_thread->error_p = NULL;
  242. new_thread->sll = NULL;
  243. pthread_sched_prevent();
  244. pthread_sched_other_resume(new_thread);
  245. /*
  246.          * Assignment must be outside of the locked pthread kernel incase
  247.          * thread is a bogus address resulting in a seg-fault. We want the
  248.          * original thread to be capable of handling the resulting signal.
  249.          * --proven
  250.          */
  251. (*thread) = new_thread;
  252. } else {
  253. retval = EAGAIN;
  254. }
  255. return(retval);
  256. }