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

MySQL数据库

开发平台:

Visual C++

  1. /* ==== pthread_join.c =======================================================
  2.  * Copyright (c) 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_join function.
  33.  *
  34.  *  1.00 94/01/15 proven
  35.  *      -Started coding this file.
  36.  */
  37. #ifndef lint
  38. static const char rcsid[] = "$Id$";
  39. #endif
  40. #include <pthread.h>
  41. #include <errno.h>
  42. static int testDeadlock( struct pthread_queue *queue, pthread_t target );
  43. /* ==========================================================================
  44.  * pthread_join()
  45.  */
  46. int pthread_join(pthread_t pthread, void **thread_return)
  47. {
  48. int ret;
  49. pthread_sched_prevent();
  50. /* Ensure they gave us a legal pthread pointer */
  51. if( ! __pthread_is_valid( pthread ) ) {
  52. pthread_sched_resume();
  53. return(EINVAL);
  54. /* Check that thread isn't detached already */
  55. if (pthread->attr.flags & PTHREAD_DETACHED) {
  56. pthread_sched_resume();
  57. return(ESRCH);
  58. /*
  59.  * Now check if other thread has exited
  60.  * Note: This must happen after checking detached state.
  61.  */
  62. if (pthread_queue_remove(&pthread_dead_queue, pthread) != OK) {
  63. /* Before we pend on the join, ensure there is no dead lock */
  64. if( testDeadlock( &pthread_run->join_queue, pthread ) == NOTOK ) {
  65. ret = EDEADLK;
  66. } else {
  67. pthread_queue_enq(&(pthread->join_queue), pthread_run);
  68. SET_PF_AT_CANCEL_POINT(pthread_run); /* This is a cancel point */
  69. pthread_resched_resume(PS_JOIN);
  70. CLEAR_PF_AT_CANCEL_POINT(pthread_run); /* No longer at cancel point */
  71. pthread_sched_prevent();
  72. if (pthread_queue_remove(&pthread_dead_queue, pthread) == OK) {
  73. pthread_queue_enq(&pthread_alloc_queue, pthread);
  74. pthread->attr.flags |= PTHREAD_DETACHED;
  75. pthread->state = PS_UNALLOCED;
  76. if (thread_return) {
  77. *thread_return = pthread->ret;
  78. }
  79. ret = OK;
  80. } else {
  81. ret = ESRCH;
  82. }
  83. }
  84.     } else {
  85. /* Just get the return value and detach the thread */
  86. pthread_queue_enq(&pthread_alloc_queue, pthread);
  87. pthread->attr.flags |= PTHREAD_DETACHED;
  88. pthread->state = PS_UNALLOCED;
  89. if (thread_return) {
  90.             *thread_return = pthread->ret;
  91.         }
  92. ret = OK;
  93. }
  94. pthread_sched_resume();
  95. return(ret);
  96. }
  97. /*----------------------------------------------------------------------
  98.  * Function: testDeadlock
  99.  * Purpose: recursive queue walk to check for deadlocks
  100.  * Args:
  101.  * queue = the queue to walk
  102.  * pthread = target to scan for
  103.  * Returns:
  104.  * OK = no deadlock, NOTOK = deadlock
  105.  * Notes:
  106.  *----------------------------------------------------------------------*/
  107. static int
  108. testDeadlock( struct pthread_queue *queue, pthread_t target )
  109. {
  110. pthread_t t;
  111. if( queue == NULL )
  112. return OK; /* Empty queue, obviously ok */
  113. for( t = queue->q_next; t; t = t->next ) {
  114. if( t == target )
  115. return NOTOK; /* bang, your dead */
  116. if( testDeadlock( &t->join_queue, target ) == NOTOK ) {
  117. return NOTOK;
  118. }
  119. }
  120. return OK; /* No deadlock */
  121. }