Main Page | Alphabetical List | Data Structures | Directories | File List | Data Fields | Globals

gw-rwlock.c

Go to the documentation of this file.
00001 /* ==================================================================== 
00002  * The Kannel Software License, Version 1.0 
00003  * 
00004  * Copyright (c) 2001-2008 Kannel Group  
00005  * Copyright (c) 1998-2001 WapIT Ltd.   
00006  * All rights reserved. 
00007  * 
00008  * Redistribution and use in source and binary forms, with or without 
00009  * modification, are permitted provided that the following conditions 
00010  * are met: 
00011  * 
00012  * 1. Redistributions of source code must retain the above copyright 
00013  *    notice, this list of conditions and the following disclaimer. 
00014  * 
00015  * 2. Redistributions in binary form must reproduce the above copyright 
00016  *    notice, this list of conditions and the following disclaimer in 
00017  *    the documentation and/or other materials provided with the 
00018  *    distribution. 
00019  * 
00020  * 3. The end-user documentation included with the redistribution, 
00021  *    if any, must include the following acknowledgment: 
00022  *       "This product includes software developed by the 
00023  *        Kannel Group (http://www.kannel.org/)." 
00024  *    Alternately, this acknowledgment may appear in the software itself, 
00025  *    if and wherever such third-party acknowledgments normally appear. 
00026  * 
00027  * 4. The names "Kannel" and "Kannel Group" must not be used to 
00028  *    endorse or promote products derived from this software without 
00029  *    prior written permission. For written permission, please  
00030  *    contact org@kannel.org. 
00031  * 
00032  * 5. Products derived from this software may not be called "Kannel", 
00033  *    nor may "Kannel" appear in their name, without prior written 
00034  *    permission of the Kannel Group. 
00035  * 
00036  * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED 
00037  * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES 
00038  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE 
00039  * DISCLAIMED.  IN NO EVENT SHALL THE KANNEL GROUP OR ITS CONTRIBUTORS 
00040  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY,  
00041  * OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT  
00042  * OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR  
00043  * BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,  
00044  * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE  
00045  * OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,  
00046  * EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 
00047  * ==================================================================== 
00048  * 
00049  * This software consists of voluntary contributions made by many 
00050  * individuals on behalf of the Kannel Group.  For more information on  
00051  * the Kannel Group, please see <http://www.kannel.org/>. 
00052  * 
00053  * Portions of this software are based upon software originally written at  
00054  * WapIT Ltd., Helsinki, Finland for the Kannel project.  
00055  */ 
00056 
00057 /*
00058  * gw-rwlock.c: Implements Reader/Writer Lock.
00059  * If pthread_rwlock_XXX functions are present then those will be used;
00060  * otherwise emulation with mutexes/condition should be done.
00061  *
00062  * Alexander Malysh <amalysh@kannel.org>, initial version 2004
00063  */
00064 
00065 #include "gw-config.h"
00066 #include "gwlib.h"
00067 #include "gw-rwlock.h"
00068 
00069 #define DEBUG 0
00070 #if DEBUG
00071   #define RWDEBUG(str,lev,frm,args...) debug(str, lev, frm, ## args)
00072 #else
00073   #define RWDEBUG(str,lev,frm,args...) do{}while(0)
00074 #endif
00075 
00076 
00077 RWLock *gw_rwlock_create(void)
00078 {
00079     RWLock *ret = gw_malloc(sizeof(*ret));
00080 #ifdef HAVE_PTHREAD_RWLOCK
00081     int rc = pthread_rwlock_init(&ret->rwlock, NULL);
00082     if (rc != 0)
00083         panic(rc, "Initialization of RWLock failed.");
00084 #else
00085     ret->writer = -1;
00086     ret->rwlock = gwlist_create();
00087     if (ret->rwlock == NULL)
00088         panic(0, "Initialization of RWLock failed.");
00089 #endif
00090     ret->dynamic = 1;
00091 
00092     return ret;
00093 }
00094 
00095 
00096 void gw_rwlock_init_static(RWLock *lock)
00097 {
00098 #ifdef HAVE_PTHREAD_RWLOCK
00099     int rc = pthread_rwlock_init(&lock->rwlock, NULL);
00100     if (rc != 0)
00101         panic(rc, "Initialization of RWLock failed.");
00102 #else
00103     lock->writer = -1;
00104     lock->rwlock = gwlist_create();
00105     if (lock->rwlock == NULL)
00106         panic(0, "Initialization of RWLock failed.");
00107 #endif
00108     lock->dynamic = 0;
00109 }
00110 
00111 
00112 void gw_rwlock_destroy(RWLock *lock)
00113 {
00114 #ifdef HAVE_PTHREAD_RWLOCK
00115     int ret;
00116 #endif
00117 
00118     if (!lock)
00119         return;
00120 
00121 #ifdef HAVE_PTHREAD_RWLOCK
00122     ret = pthread_rwlock_destroy(&lock->rwlock);
00123     if (ret != 0)
00124         panic(ret, "Attempt to destroy locked rwlock.");
00125 #else
00126     gwlist_destroy(lock->rwlock, NULL);
00127 #endif
00128 
00129     if (lock->dynamic)
00130         gw_free(lock);
00131 }
00132 
00133 
00134 int gw_rwlock_rdlock(RWLock *lock)
00135 {
00136     int ret = 0;
00137     gw_assert(lock != NULL);
00138 
00139 #ifdef HAVE_PTHREAD_RWLOCK
00140     ret = pthread_rwlock_rdlock(&lock->rwlock);
00141     if (ret != 0) {
00142         panic(ret, "Error while pthread_rwlock_rdlock.");
00143     }
00144 #else
00145     gwlist_lock(lock->rwlock);
00146     gwlist_add_producer(lock->rwlock);
00147     gwlist_unlock(lock->rwlock);
00148     RWDEBUG("", 0, "------------ gw_rwlock_rdlock(%p) ----------", lock);
00149 #endif
00150 
00151     return ret;
00152 }
00153 
00154 
00155 int gw_rwlock_unlock(RWLock *lock)
00156 {
00157     int ret = 0;
00158     gw_assert(lock != NULL);
00159 
00160 #ifdef HAVE_PTHREAD_RWLOCK
00161     ret = pthread_rwlock_unlock(&lock->rwlock);
00162     if (ret != 0)
00163         panic(ret, "Error while gw_rwlock_unlock.");
00164 #else
00165     RWDEBUG("", 0, "------------ gw_rwlock_unlock(%p) ----------", lock);
00166     if (lock->writer == gwthread_self()) {
00167         lock->writer = -1;
00168         gwlist_unlock(lock->rwlock);
00169     } else 
00170         gwlist_remove_producer(lock->rwlock);
00171 #endif
00172 
00173     return ret;
00174 }
00175 
00176 
00177 int gw_rwlock_wrlock(RWLock *lock)
00178 {
00179     int ret = 0;
00180     gw_assert(lock != NULL);
00181 
00182 #ifdef HAVE_PTHREAD_RWLOCK
00183     ret = pthread_rwlock_wrlock(&lock->rwlock);
00184     if (ret != 0)
00185         panic(ret, "Error while pthread_rwlock_wrlock.");
00186 #else
00187     RWDEBUG("", 0, "------------ gw_rwlock_wrlock(%p) ----------", lock);
00188     gwlist_lock(lock->rwlock);
00189     RWDEBUG("", 0, "------------ gw_rwlock_wrlock(%p) producers=%d", lock, gwlist_producer_count(lock->rwlock));
00190     /* wait for reader */
00191     gwlist_consume(lock->rwlock);
00192     lock->writer = gwthread_self();
00193 #endif
00194 
00195     return ret;
00196 }
See file LICENSE for details about the license agreement for using, modifying, copying or deriving work from this software.