fwalk.c
Upload User: tsgydb
Upload Date: 2007-04-14
Package Size: 10674k
Code Size: 5k
Category:

MySQL

Development Platform:

Visual C++

  1. /*-
  2.  * Copyright (c) 1990 The Regents of the University of California.
  3.  * Copyright (c) 1993, 1994 Chris Provenzano. 
  4.  * All rights reserved.
  5.  *
  6.  * This code is derived from software contributed to Berkeley by
  7.  * Chris Torek.
  8.  *
  9.  * Redistribution and use in source and binary forms, with or without
  10.  * modification, are permitted provided that the following conditions
  11.  * are met:
  12.  * 1. Redistributions of source code must retain the above copyright
  13.  *    notice, this list of conditions and the following disclaimer.
  14.  * 2. Redistributions in binary form must reproduce the above copyright
  15.  *    notice, this list of conditions and the following disclaimer in the
  16.  *    documentation and/or other materials provided with the distribution.
  17.  * 3. All advertising materials mentioning features or use of this software
  18.  *    must display the following acknowledgement:
  19.  * This product includes software developed by the University of
  20.  * California, Berkeley and its contributors.
  21.  * 4. Neither the name of the University nor the names of its contributors
  22.  *    may be used to endorse or promote products derived from this software
  23.  *    without specific prior written permission.
  24.  *
  25.  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
  26.  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  27.  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  28.  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
  29.  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
  30.  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
  31.  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
  32.  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
  33.  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
  34.  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
  35.  * SUCH DAMAGE.
  36.  */
  37. #if defined(LIBC_SCCS) && !defined(lint)
  38. /*static char *sccsid = "from: @(#)fwalk.c 5.2 (Berkeley) 2/24/91";*/
  39. static char *rcsid = "$Id$";
  40. #endif /* LIBC_SCCS and not lint */
  41. #include <pthread.h>
  42. #include <errno.h>
  43. #include <stdio.h>
  44. #include "local.h"
  45. #include "glue.h"
  46. extern pthread_mutex_t __sfp_mutex;
  47. extern pthread_cond_t __sfp_cond;
  48. extern int __sfp_state;
  49. /*
  50.  * fwalk now can only be used for flushing the buffers.
  51.  * This is all it was originally used for.
  52.  * The function has also become much more complicated.
  53.  * The first time through we flush everything we can.
  54.  * If this fails to flush everything because we couldn't get a lock
  55.  * we wait on the locksfor the second pass. Why this works ...
  56.  * 
  57.  * This function must allow for multiple threads to flush everything.
  58.  * This function cannot flush buffers locked by another thread.
  59.  * So we flush everything we can the first pass. This includes all
  60.  * buffers locked by this thread, and wait on buffers that are locked.
  61.  * Eventually other threads willl unlock there buffers or flush them themselves
  62.  * at which point this thread will notice that it's empty or be able to
  63.  * flush the buffer. This is fine so long as no other thread tries to flush
  64.  * all buffers. Here is the possible deadlock condition, but since this thread
  65.  * has flushed all buffers it can, there are NO buffers locked by this thread
  66.  * that need flushing so any other thread flushing won't block waiting on this
  67.  * thread thereby eliminating the deadlock condition.
  68.  */
  69. int __swalk_sflush()
  70. {
  71. register FILE *fp, *savefp;
  72. register int n, ret, saven;
  73. register struct glue *g, *saveg;
  74. /* Only allow other threads to read __sglue */
  75. pthread_mutex_lock(&__sfp_mutex);
  76. __sfp_state++;
  77. pthread_mutex_unlock(&__sfp_mutex);
  78. ret = 0;
  79. saven = 0;
  80. saveg = NULL;
  81. savefp = NULL;
  82. for (g = &__sglue; g != NULL; g = g->next) {
  83. for (fp = g->iobs, n = g->niobs; --n >= 0; fp++) {
  84. if (fp->_flags != 0) {
  85. /* Is there anything to flush? */
  86. if (fp->_bf._base && (fp->_bf._base - fp->_p)) {
  87. if (ftrylockfile(fp)) { /* Can we flush it */
  88. if (!saven) { /* No, save first fp we can't flush */
  89. saven = n;
  90. saveg = g;
  91. savefp = fp;
  92. continue;
  93. }
  94. } else {
  95. ret |= __sflush(fp);
  96. funlockfile(fp);
  97. }
  98. }
  99. }
  100. }
  101. }
  102. if (savefp) {
  103. for (g = saveg; g != NULL; g = g->next) {
  104. for (fp = savefp, n = saven + 1; --n >= 0; fp++) {
  105. if (fp->_flags != 0) {
  106.   /* Anything to flush */
  107. while (fp->_bf._base && (fp->_bf._base - fp->_p)) {
  108. flockfile(fp);
  109. ret |= __sflush(fp);
  110. funlockfile(fp);
  111. }
  112. }
  113. }
  114. }
  115. }
  116. /* If no other readers wakeup a thread waiting to do __sfp */
  117. pthread_mutex_lock(&__sfp_mutex);
  118. if (! (--__sfp_state)) {
  119. pthread_cond_signal(&__sfp_cond);
  120. }
  121. pthread_mutex_unlock(&__sfp_mutex);
  122. return (ret);
  123. }