/* Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. * The ASF licenses this file to You under the Apache License, Version 2.0 * (the "License"); you may not use this file except in compliance with * the License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "apr.h" #include "apr_strings.h" #include "apr_arch_proc_mutex.h" #include "apr_arch_file_io.h" /* for apr_mkstemp() */ #include "apr_hash.h" #include "apr_atomic.h" APR_DECLARE(apr_status_t) apr_proc_mutex_destroy(apr_proc_mutex_t *mutex) { return apr_pool_cleanup_run(mutex->pool, mutex, apr_proc_mutex_cleanup); } #if APR_HAS_POSIXSEM_SERIALIZE || APR_HAS_FCNTL_SERIALIZE || \ APR_HAS_SYSVSEM_SERIALIZE static apr_status_t proc_mutex_no_child_init(apr_proc_mutex_t **mutex, apr_pool_t *cont, const char *fname) { return APR_SUCCESS; } #endif #if APR_HAS_POSIXSEM_SERIALIZE || APR_HAS_PROC_PTHREAD_SERIALIZE static apr_status_t proc_mutex_no_perms_set(apr_proc_mutex_t *mutex, apr_fileperms_t perms, apr_uid_t uid, apr_gid_t gid) { return APR_ENOTIMPL; } #endif #if APR_HAS_FCNTL_SERIALIZE \ || APR_HAS_FLOCK_SERIALIZE \ || (APR_HAS_SYSVSEM_SERIALIZE \ && !defined(HAVE_SEMTIMEDOP)) \ || (APR_HAS_POSIXSEM_SERIALIZE \ && !defined(HAVE_SEM_TIMEDWAIT)) \ || (APR_HAS_PROC_PTHREAD_SERIALIZE \ && !defined(HAVE_PTHREAD_MUTEX_TIMEDLOCK) \ && !defined(HAVE_PTHREAD_CONDATTR_SETPSHARED)) static apr_status_t proc_mutex_spinsleep_timedacquire(apr_proc_mutex_t *mutex, apr_interval_time_t timeout) { #define SLEEP_TIME apr_time_from_msec(10) apr_status_t rv; for (;;) { rv = apr_proc_mutex_trylock(mutex); if (!APR_STATUS_IS_EBUSY(rv)) { if (rv == APR_SUCCESS) { mutex->curr_locked = 1; } break; } if (timeout <= 0) { rv = APR_TIMEUP; break; } if (timeout > SLEEP_TIME) { apr_sleep(SLEEP_TIME); timeout -= SLEEP_TIME; } else { apr_sleep(timeout); timeout = 0; } } return rv; } #endif #if APR_HAS_POSIXSEM_SERIALIZE #ifndef SEM_FAILED #define SEM_FAILED (-1) #endif static apr_status_t proc_mutex_posix_cleanup(void *mutex_) { apr_proc_mutex_t *mutex = mutex_; if (sem_close(mutex->os.psem_interproc) < 0) { return errno; } return APR_SUCCESS; } static unsigned int rshash (char *p) { /* hash function from Robert Sedgwicks 'Algorithms in C' book */ unsigned int b = 378551; unsigned int a = 63689; unsigned int retval = 0; for( ; *p; p++) { retval = retval * a + (*p); a *= b; } return retval; } static apr_status_t proc_mutex_posix_create(apr_proc_mutex_t *new_mutex, const char *fname) { #define APR_POSIXSEM_NAME_MIN 13 sem_t *psem; char semname[32]; /* * This bogusness is to follow what appears to be the * lowest common denominator in Posix semaphore naming: * - start with '/' * - be at most 14 chars * - be unique and not match anything on the filesystem * * Because of this, we use fname to generate a (unique) hash * and use that as the name of the semaphore. If no filename was * given, we create one based on the time. We tuck the name * away, since it might be useful for debugging. We use 2 hashing * functions to try to avoid collisions. * * To make this as robust as possible, we initially try something * larger (and hopefully more unique) and gracefully fail down to the * LCD above. * * NOTE: Darwin (Mac OS X) seems to be the most restrictive * implementation. Versions previous to Darwin 6.2 had the 14 * char limit, but later rev's allow up to 31 characters. * */ if (fname) { apr_ssize_t flen = strlen(fname); char *p = apr_pstrndup(new_mutex->pool, fname, strlen(fname)); unsigned int h1, h2; h1 = (apr_hashfunc_default((const char *)p, &flen) & 0xffffffff); h2 = (rshash(p) & 0xffffffff); apr_snprintf(semname, sizeof(semname), "/ApR.%xH%x", h1, h2); } else { apr_time_t now; unsigned long sec; unsigned long usec; now = apr_time_now(); sec = apr_time_sec(now); usec = apr_time_usec(now); apr_snprintf(semname, sizeof(semname), "/ApR.%lxZ%lx", sec, usec); } do { psem = sem_open(semname, O_CREAT | O_EXCL, 0644, 1); } while (psem == (sem_t *)SEM_FAILED && errno == EINTR); if (psem == (sem_t *)SEM_FAILED) { if (errno == ENAMETOOLONG) { /* Oh well, good try */ semname[APR_POSIXSEM_NAME_MIN] = '\0'; } else { return errno; } do { psem = sem_open(semname, O_CREAT | O_EXCL, 0644, 1); } while (psem == (sem_t *)SEM_FAILED && errno == EINTR); } if (psem == (sem_t *)SEM_FAILED) { return errno; } /* Ahhh. The joys of Posix sems. Predelete it... */ sem_unlink(semname); new_mutex->os.psem_interproc = psem; new_mutex->fname = apr_pstrdup(new_mutex->pool, semname); apr_pool_cleanup_register(new_mutex->pool, (void *)new_mutex, apr_proc_mutex_cleanup, apr_pool_cleanup_null); return APR_SUCCESS; } static apr_status_t proc_mutex_posix_acquire(apr_proc_mutex_t *mutex) { int rc; do { rc = sem_wait(mutex->os.psem_interproc); } while (rc < 0 && errno == EINTR); if (rc < 0) { return errno; } mutex->curr_locked = 1; return APR_SUCCESS; } static apr_status_t proc_mutex_posix_tryacquire(apr_proc_mutex_t *mutex) { int rc; do { rc = sem_trywait(mutex->os.psem_interproc); } while (rc < 0 && errno == EINTR); if (rc < 0) { if (errno == EAGAIN) { return APR_EBUSY; } return errno; } mutex->curr_locked = 1; return APR_SUCCESS; } #if defined(HAVE_SEM_TIMEDWAIT) static apr_status_t proc_mutex_posix_timedacquire(apr_proc_mutex_t *mutex, apr_interval_time_t timeout) { if (timeout <= 0) { apr_status_t rv = proc_mutex_posix_tryacquire(mutex); return (rv == APR_EBUSY) ? APR_TIMEUP : rv; } else { int rc; struct timespec abstime; timeout += apr_time_now(); abstime.tv_sec = apr_time_sec(timeout); abstime.tv_nsec = apr_time_usec(timeout) * 1000; /* nanoseconds */ do { rc = sem_timedwait(mutex->os.psem_interproc, &abstime); } while (rc < 0 && errno == EINTR); if (rc < 0) { if (errno == ETIMEDOUT) { return APR_TIMEUP; } return errno; } } mutex->curr_locked = 1; return APR_SUCCESS; } #endif static apr_status_t proc_mutex_posix_release(apr_proc_mutex_t *mutex) { mutex->curr_locked = 0; if (sem_post(mutex->os.psem_interproc) < 0) { /* any failure is probably fatal, so no big deal to leave * ->curr_locked at 0. */ return errno; } return APR_SUCCESS; } static const apr_proc_mutex_unix_lock_methods_t mutex_posixsem_methods = { #if APR_PROCESS_LOCK_IS_GLOBAL || !APR_HAS_THREADS || defined(POSIXSEM_IS_GLOBAL) APR_PROCESS_LOCK_MECH_IS_GLOBAL, #else 0, #endif proc_mutex_posix_create, proc_mutex_posix_acquire, proc_mutex_posix_tryacquire, #if defined(HAVE_SEM_TIMEDWAIT) proc_mutex_posix_timedacquire, #else proc_mutex_spinsleep_timedacquire, #endif proc_mutex_posix_release, proc_mutex_posix_cleanup, proc_mutex_no_child_init, proc_mutex_no_perms_set, APR_LOCK_POSIXSEM, "posixsem" }; #endif /* Posix sem implementation */ #if APR_HAS_SYSVSEM_SERIALIZE static struct sembuf proc_mutex_op_on; static struct sembuf proc_mutex_op_try; static struct sembuf proc_mutex_op_off; static void proc_mutex_sysv_setup(void) { proc_mutex_op_on.sem_num = 0; proc_mutex_op_on.sem_op = -1; proc_mutex_op_on.sem_flg = SEM_UNDO; proc_mutex_op_try.sem_num = 0; proc_mutex_op_try.sem_op = -1; proc_mutex_op_try.sem_flg = SEM_UNDO | IPC_NOWAIT; proc_mutex_op_off.sem_num = 0; proc_mutex_op_off.sem_op = 1; proc_mutex_op_off.sem_flg = SEM_UNDO; } static apr_status_t proc_mutex_sysv_cleanup(void *mutex_) { apr_proc_mutex_t *mutex=mutex_; union semun ick; if (mutex->os.crossproc != -1) { ick.val = 0; semctl(mutex->os.crossproc, 0, IPC_RMID, ick); } return APR_SUCCESS; } static apr_status_t proc_mutex_sysv_create(apr_proc_mutex_t *new_mutex, const char *fname) { union semun ick; apr_status_t rv; new_mutex->os.crossproc = semget(IPC_PRIVATE, 1, IPC_CREAT | 0600); if (new_mutex->os.crossproc == -1) { rv = errno; proc_mutex_sysv_cleanup(new_mutex); return rv; } ick.val = 1; if (semctl(new_mutex->os.crossproc, 0, SETVAL, ick) < 0) { rv = errno; proc_mutex_sysv_cleanup(new_mutex); new_mutex->os.crossproc = -1; return rv; } new_mutex->curr_locked = 0; apr_pool_cleanup_register(new_mutex->pool, (void *)new_mutex, apr_proc_mutex_cleanup, apr_pool_cleanup_null); return APR_SUCCESS; } static apr_status_t proc_mutex_sysv_acquire(apr_proc_mutex_t *mutex) { int rc; do { rc = semop(mutex->os.crossproc, &proc_mutex_op_on, 1); } while (rc < 0 && errno == EINTR); if (rc < 0) { return errno; } mutex->curr_locked = 1; return APR_SUCCESS; } static apr_status_t proc_mutex_sysv_tryacquire(apr_proc_mutex_t *mutex) { int rc; do { rc = semop(mutex->os.crossproc, &proc_mutex_op_try, 1); } while (rc < 0 && errno == EINTR); if (rc < 0) { if (errno == EAGAIN) { return APR_EBUSY; } return errno; } mutex->curr_locked = 1; return APR_SUCCESS; } #if defined(HAVE_SEMTIMEDOP) static apr_status_t proc_mutex_sysv_timedacquire(apr_proc_mutex_t *mutex, apr_interval_time_t timeout) { if (timeout <= 0) { apr_status_t rv = proc_mutex_sysv_tryacquire(mutex); return (rv == APR_EBUSY) ? APR_TIMEUP : rv; } else { int rc; struct timespec reltime; reltime.tv_sec = apr_time_sec(timeout); reltime.tv_nsec = apr_time_usec(timeout) * 1000; /* nanoseconds */ do { rc = semtimedop(mutex->os.crossproc, &proc_mutex_op_on, 1, &reltime); } while (rc < 0 && errno == EINTR); if (rc < 0) { if (errno == EAGAIN) { return APR_TIMEUP; } return errno; } } mutex->curr_locked = 1; return APR_SUCCESS; } #endif static apr_status_t proc_mutex_sysv_release(apr_proc_mutex_t *mutex) { int rc; mutex->curr_locked = 0; do { rc = semop(mutex->os.crossproc, &proc_mutex_op_off, 1); } while (rc < 0 && errno == EINTR); if (rc < 0) { return errno; } return APR_SUCCESS; } static apr_status_t proc_mutex_sysv_perms_set(apr_proc_mutex_t *mutex, apr_fileperms_t perms, apr_uid_t uid, apr_gid_t gid) { union semun ick; struct semid_ds buf; buf.sem_perm.uid = uid; buf.sem_perm.gid = gid; buf.sem_perm.mode = apr_unix_perms2mode(perms); ick.buf = &buf; if (semctl(mutex->os.crossproc, 0, IPC_SET, ick) < 0) { return errno; } return APR_SUCCESS; } static const apr_proc_mutex_unix_lock_methods_t mutex_sysv_methods = { #if APR_PROCESS_LOCK_IS_GLOBAL || !APR_HAS_THREADS || defined(SYSVSEM_IS_GLOBAL) APR_PROCESS_LOCK_MECH_IS_GLOBAL, #else 0, #endif proc_mutex_sysv_create, proc_mutex_sysv_acquire, proc_mutex_sysv_tryacquire, #if defined(HAVE_SEMTIMEDOP) proc_mutex_sysv_timedacquire, #else proc_mutex_spinsleep_timedacquire, #endif proc_mutex_sysv_release, proc_mutex_sysv_cleanup, proc_mutex_no_child_init, proc_mutex_sysv_perms_set, APR_LOCK_SYSVSEM, "sysvsem" }; #endif /* SysV sem implementation */ #if APR_HAS_PROC_PTHREAD_SERIALIZE #ifndef APR_USE_PROC_PTHREAD_MUTEX_COND #define APR_USE_PROC_PTHREAD_MUTEX_COND \ (defined(HAVE_PTHREAD_CONDATTR_SETPSHARED) \ && !defined(HAVE_PTHREAD_MUTEX_TIMEDLOCK)) #endif /* The mmap()ed pthread_interproc is the native pthread_mutex_t followed * by a refcounter to track children using it. We want to avoid calling * pthread_mutex_destroy() on the shared mutex area while it is in use by * another process, because this may mark the shared pthread_mutex_t as * invalid for everyone, including forked children (unlike "sysvsem" for * example), causing unexpected errors or deadlocks (PR 49504). So the * last process (parent or child) referencing the mutex will effectively * destroy it. */ typedef struct { #define proc_pthread_cast(m) \ ((proc_pthread_mutex_t *)(m)->os.pthread_interproc) pthread_mutex_t mutex; #define proc_pthread_mutex(m) \ (proc_pthread_cast(m)->mutex) #if APR_USE_PROC_PTHREAD_MUTEX_COND pthread_cond_t cond; #define proc_pthread_mutex_cond(m) \ (proc_pthread_cast(m)->cond) apr_int32_t cond_locked; #define proc_pthread_mutex_cond_locked(m) \ (proc_pthread_cast(m)->cond_locked) apr_uint32_t cond_num_waiters; #define proc_pthread_mutex_cond_num_waiters(m) \ (proc_pthread_cast(m)->cond_num_waiters) #define proc_pthread_mutex_is_cond(m) \ ((m)->pthread_refcounting && proc_pthread_mutex_cond_locked(m) != -1) #endif /* APR_USE_PROC_PTHREAD_MUTEX_COND */ apr_uint32_t refcount; #define proc_pthread_mutex_refcount(m) \ (proc_pthread_cast(m)->refcount) } proc_pthread_mutex_t; static APR_INLINE int proc_pthread_mutex_inc(apr_proc_mutex_t *mutex) { if (mutex->pthread_refcounting) { apr_atomic_inc32(&proc_pthread_mutex_refcount(mutex)); return 1; } return 0; } static APR_INLINE int proc_pthread_mutex_dec(apr_proc_mutex_t *mutex) { if (mutex->pthread_refcounting) { return apr_atomic_dec32(&proc_pthread_mutex_refcount(mutex)); } return 0; } static apr_status_t proc_pthread_mutex_unref(void *mutex_) { apr_proc_mutex_t *mutex=mutex_; apr_status_t rv; #if APR_USE_PROC_PTHREAD_MUTEX_COND if (proc_pthread_mutex_is_cond(mutex)) { mutex->curr_locked = 0; } else #endif /* APR_USE_PROC_PTHREAD_MUTEX_COND */ if (mutex->curr_locked == 1) { if ((rv = pthread_mutex_unlock(&proc_pthread_mutex(mutex)))) { #ifdef HAVE_ZOS_PTHREADS rv = errno; #endif return rv; } } if (!proc_pthread_mutex_dec(mutex)) { #if APR_USE_PROC_PTHREAD_MUTEX_COND if (proc_pthread_mutex_is_cond(mutex) && (rv = pthread_cond_destroy(&proc_pthread_mutex_cond(mutex)))) { #ifdef HAVE_ZOS_PTHREADS rv = errno; #endif return rv; } #endif /* APR_USE_PROC_PTHREAD_MUTEX_COND */ if ((rv = pthread_mutex_destroy(&proc_pthread_mutex(mutex)))) { #ifdef HAVE_ZOS_PTHREADS rv = errno; #endif return rv; } } return APR_SUCCESS; } static apr_status_t proc_mutex_pthread_cleanup(void *mutex_) { apr_proc_mutex_t *mutex=mutex_; apr_status_t rv; /* curr_locked is set to -1 until the mutex has been created */ if (mutex->curr_locked != -1) { if ((rv = proc_pthread_mutex_unref(mutex))) { return rv; } } if (munmap(mutex->os.pthread_interproc, sizeof(proc_pthread_mutex_t))) { return errno; } return APR_SUCCESS; } static apr_status_t proc_mutex_pthread_create(apr_proc_mutex_t *new_mutex, const char *fname) { apr_status_t rv; int fd; pthread_mutexattr_t mattr; fd = open("/dev/zero", O_RDWR); if (fd < 0) { return errno; } new_mutex->os.pthread_interproc = mmap(NULL, sizeof(proc_pthread_mutex_t), PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0); if (new_mutex->os.pthread_interproc == MAP_FAILED) { new_mutex->os.pthread_interproc = NULL; rv = errno; close(fd); return rv; } close(fd); new_mutex->pthread_refcounting = 1; new_mutex->curr_locked = -1; /* until the mutex has been created */ #if APR_USE_PROC_PTHREAD_MUTEX_COND proc_pthread_mutex_cond_locked(new_mutex) = -1; #endif if ((rv = pthread_mutexattr_init(&mattr))) { #ifdef HAVE_ZOS_PTHREADS rv = errno; #endif proc_mutex_pthread_cleanup(new_mutex); return rv; } if ((rv = pthread_mutexattr_setpshared(&mattr, PTHREAD_PROCESS_SHARED))) { #ifdef HAVE_ZOS_PTHREADS rv = errno; #endif proc_mutex_pthread_cleanup(new_mutex); pthread_mutexattr_destroy(&mattr); return rv; } #if defined(HAVE_PTHREAD_MUTEX_ROBUST) || defined(HAVE_PTHREAD_MUTEX_ROBUST_NP) #ifdef HAVE_PTHREAD_MUTEX_ROBUST rv = pthread_mutexattr_setrobust(&mattr, PTHREAD_MUTEX_ROBUST); #else rv = pthread_mutexattr_setrobust_np(&mattr, PTHREAD_MUTEX_ROBUST_NP); #endif if (rv) { #ifdef HAVE_ZOS_PTHREADS rv = errno; #endif proc_mutex_pthread_cleanup(new_mutex); pthread_mutexattr_destroy(&mattr); return rv; } if ((rv = pthread_mutexattr_setprotocol(&mattr, PTHREAD_PRIO_INHERIT))) { #ifdef HAVE_ZOS_PTHREADS rv = errno; #endif proc_mutex_pthread_cleanup(new_mutex); pthread_mutexattr_destroy(&mattr); return rv; } #endif /* HAVE_PTHREAD_MUTEX_ROBUST[_NP] */ if ((rv = pthread_mutex_init(&proc_pthread_mutex(new_mutex), &mattr))) { #ifdef HAVE_ZOS_PTHREADS rv = errno; #endif proc_mutex_pthread_cleanup(new_mutex); pthread_mutexattr_destroy(&mattr); return rv; } proc_pthread_mutex_refcount(new_mutex) = 1; /* first/parent reference */ new_mutex->curr_locked = 0; /* mutex created now */ if ((rv = pthread_mutexattr_destroy(&mattr))) { #ifdef HAVE_ZOS_PTHREADS rv = errno; #endif proc_mutex_pthread_cleanup(new_mutex); return rv; } apr_pool_cleanup_register(new_mutex->pool, (void *)new_mutex, apr_proc_mutex_cleanup, apr_pool_cleanup_null); return APR_SUCCESS; } static apr_status_t proc_mutex_pthread_child_init(apr_proc_mutex_t **mutex, apr_pool_t *pool, const char *fname) { (*mutex)->curr_locked = 0; if (proc_pthread_mutex_inc(*mutex)) { apr_pool_cleanup_register(pool, *mutex, proc_pthread_mutex_unref, apr_pool_cleanup_null); } return APR_SUCCESS; } static apr_status_t proc_mutex_pthread_acquire_ex(apr_proc_mutex_t *mutex, apr_interval_time_t timeout) { apr_status_t rv; #if APR_USE_PROC_PTHREAD_MUTEX_COND if (proc_pthread_mutex_is_cond(mutex)) { if ((rv = pthread_mutex_lock(&proc_pthread_mutex(mutex)))) { #ifdef HAVE_ZOS_PTHREADS rv = errno; #endif #if defined(HAVE_PTHREAD_MUTEX_ROBUST) || defined(HAVE_PTHREAD_MUTEX_ROBUST_NP) /* Okay, our owner died. Let's try to make it consistent again. */ if (rv == EOWNERDEAD) { proc_pthread_mutex_dec(mutex); #ifdef HAVE_PTHREAD_MUTEX_ROBUST pthread_mutex_consistent(&proc_pthread_mutex(mutex)); #else pthread_mutex_consistent_np(&proc_pthread_mutex(mutex)); #endif } else #endif return rv; } if (!proc_pthread_mutex_cond_locked(mutex)) { rv = APR_SUCCESS; } else if (!timeout) { rv = APR_TIMEUP; } else { struct timespec abstime; if (timeout > 0) { timeout += apr_time_now(); abstime.tv_sec = apr_time_sec(timeout); abstime.tv_nsec = apr_time_usec(timeout) * 1000; /* nanoseconds */ } proc_pthread_mutex_cond_num_waiters(mutex)++; do { if (timeout < 0) { rv = pthread_cond_wait(&proc_pthread_mutex_cond(mutex), &proc_pthread_mutex(mutex)); if (rv) { #ifdef HAVE_ZOS_PTHREADS rv = errno; #endif break; } } else { rv = pthread_cond_timedwait(&proc_pthread_mutex_cond(mutex), &proc_pthread_mutex(mutex), &abstime); if (rv) { #ifdef HAVE_ZOS_PTHREADS rv = errno; #endif if (rv == ETIMEDOUT) { rv = APR_TIMEUP; } break; } } } while (proc_pthread_mutex_cond_locked(mutex)); proc_pthread_mutex_cond_num_waiters(mutex)--; } if (rv != APR_SUCCESS) { pthread_mutex_unlock(&proc_pthread_mutex(mutex)); return rv; } proc_pthread_mutex_cond_locked(mutex) = 1; rv = pthread_mutex_unlock(&proc_pthread_mutex(mutex)); if (rv) { #ifdef HAVE_ZOS_PTHREADS rv = errno; #endif return rv; } } else #endif /* APR_USE_PROC_PTHREAD_MUTEX_COND */ { if (timeout < 0) { rv = pthread_mutex_lock(&proc_pthread_mutex(mutex)); if (rv) { #ifdef HAVE_ZOS_PTHREADS rv = errno; #endif } } else if (!timeout) { rv = pthread_mutex_trylock(&proc_pthread_mutex(mutex)); if (rv) { #ifdef HAVE_ZOS_PTHREADS rv = errno; #endif if (rv == EBUSY) { return APR_TIMEUP; } } } else #if defined(HAVE_PTHREAD_MUTEX_TIMEDLOCK) { struct timespec abstime; timeout += apr_time_now(); abstime.tv_sec = apr_time_sec(timeout); abstime.tv_nsec = apr_time_usec(timeout) * 1000; /* nanoseconds */ rv = pthread_mutex_timedlock(&proc_pthread_mutex(mutex), &abstime); if (rv) { #ifdef HAVE_ZOS_PTHREADS rv = errno; #endif if (rv == ETIMEDOUT) { return APR_TIMEUP; } } } if (rv) { #if defined(HAVE_PTHREAD_MUTEX_ROBUST) || defined(HAVE_PTHREAD_MUTEX_ROBUST_NP) /* Okay, our owner died. Let's try to make it consistent again. */ if (rv == EOWNERDEAD) { proc_pthread_mutex_dec(mutex); #ifdef HAVE_PTHREAD_MUTEX_ROBUST pthread_mutex_consistent(&proc_pthread_mutex(mutex)); #else pthread_mutex_consistent_np(&proc_pthread_mutex(mutex)); #endif } else #endif return rv; } #else /* !HAVE_PTHREAD_MUTEX_TIMEDLOCK */ return proc_mutex_spinsleep_timedacquire(mutex, timeout); #endif } mutex->curr_locked = 1; return APR_SUCCESS; } static apr_status_t proc_mutex_pthread_acquire(apr_proc_mutex_t *mutex) { return proc_mutex_pthread_acquire_ex(mutex, -1); } static apr_status_t proc_mutex_pthread_tryacquire(apr_proc_mutex_t *mutex) { apr_status_t rv = proc_mutex_pthread_acquire_ex(mutex, 0); return (rv == APR_TIMEUP) ? APR_EBUSY : rv; } static apr_status_t proc_mutex_pthread_timedacquire(apr_proc_mutex_t *mutex, apr_interval_time_t timeout) { return proc_mutex_pthread_acquire_ex(mutex, (timeout <= 0) ? 0 : timeout); } static apr_status_t proc_mutex_pthread_release(apr_proc_mutex_t *mutex) { apr_status_t rv; #if APR_USE_PROC_PTHREAD_MUTEX_COND if (proc_pthread_mutex_is_cond(mutex)) { if ((rv = pthread_mutex_lock(&proc_pthread_mutex(mutex)))) { #ifdef HAVE_ZOS_PTHREADS rv = errno; #endif #if defined(HAVE_PTHREAD_MUTEX_ROBUST) || defined(HAVE_PTHREAD_MUTEX_ROBUST_NP) /* Okay, our owner died. Let's try to make it consistent again. */ if (rv == EOWNERDEAD) { proc_pthread_mutex_dec(mutex); #ifdef HAVE_PTHREAD_MUTEX_ROBUST pthread_mutex_consistent(&proc_pthread_mutex(mutex)); #else pthread_mutex_consistent_np(&proc_pthread_mutex(mutex)); #endif } else #endif return rv; } if (!proc_pthread_mutex_cond_locked(mutex)) { rv = APR_EINVAL; } else if (!proc_pthread_mutex_cond_num_waiters(mutex)) { rv = APR_SUCCESS; } else { rv = pthread_cond_signal(&proc_pthread_mutex_cond(mutex)); #ifdef HAVE_ZOS_PTHREADS if (rv) { rv = errno; } #endif } if (rv != APR_SUCCESS) { pthread_mutex_unlock(&proc_pthread_mutex(mutex)); return rv; } proc_pthread_mutex_cond_locked(mutex) = 0; } #endif /* APR_USE_PROC_PTHREAD_MUTEX_COND */ mutex->curr_locked = 0; if ((rv = pthread_mutex_unlock(&proc_pthread_mutex(mutex)))) { #ifdef HAVE_ZOS_PTHREADS rv = errno; #endif return rv; } return APR_SUCCESS; } static const apr_proc_mutex_unix_lock_methods_t mutex_proc_pthread_methods = { APR_PROCESS_LOCK_MECH_IS_GLOBAL, proc_mutex_pthread_create, proc_mutex_pthread_acquire, proc_mutex_pthread_tryacquire, proc_mutex_pthread_timedacquire, proc_mutex_pthread_release, proc_mutex_pthread_cleanup, proc_mutex_pthread_child_init, proc_mutex_no_perms_set, APR_LOCK_PROC_PTHREAD, "pthread" }; #if APR_USE_PROC_PTHREAD_MUTEX_COND static apr_status_t proc_mutex_pthread_cond_create(apr_proc_mutex_t *new_mutex, const char *fname) { apr_status_t rv; pthread_condattr_t cattr; rv = proc_mutex_pthread_create(new_mutex, fname); if (rv != APR_SUCCESS) { return rv; } if ((rv = pthread_condattr_init(&cattr))) { #ifdef HAVE_ZOS_PTHREADS rv = errno; #endif apr_pool_cleanup_run(new_mutex->pool, new_mutex, apr_proc_mutex_cleanup); return rv; } if ((rv = pthread_condattr_setpshared(&cattr, PTHREAD_PROCESS_SHARED))) { #ifdef HAVE_ZOS_PTHREADS rv = errno; #endif pthread_condattr_destroy(&cattr); apr_pool_cleanup_run(new_mutex->pool, new_mutex, apr_proc_mutex_cleanup); return rv; } if ((rv = pthread_cond_init(&proc_pthread_mutex_cond(new_mutex), &cattr))) { #ifdef HAVE_ZOS_PTHREADS rv = errno; #endif pthread_condattr_destroy(&cattr); apr_pool_cleanup_run(new_mutex->pool, new_mutex, apr_proc_mutex_cleanup); return rv; } pthread_condattr_destroy(&cattr); proc_pthread_mutex_cond_locked(new_mutex) = 0; proc_pthread_mutex_cond_num_waiters(new_mutex) = 0; return APR_SUCCESS; } static const apr_proc_mutex_unix_lock_methods_t mutex_proc_pthread_cond_methods = { APR_PROCESS_LOCK_MECH_IS_GLOBAL, proc_mutex_pthread_cond_create, proc_mutex_pthread_acquire, proc_mutex_pthread_tryacquire, proc_mutex_pthread_timedacquire, proc_mutex_pthread_release, proc_mutex_pthread_cleanup, proc_mutex_pthread_child_init, proc_mutex_no_perms_set, APR_LOCK_PROC_PTHREAD, "pthread" }; #endif #endif #if APR_HAS_FCNTL_SERIALIZE static struct flock proc_mutex_lock_it; static struct flock proc_mutex_unlock_it; static apr_status_t proc_mutex_fcntl_release(apr_proc_mutex_t *); static void proc_mutex_fcntl_setup(void) { proc_mutex_lock_it.l_whence = SEEK_SET; /* from current point */ proc_mutex_lock_it.l_start = 0; /* -"- */ proc_mutex_lock_it.l_len = 0; /* until end of file */ proc_mutex_lock_it.l_type = F_WRLCK; /* set exclusive/write lock */ proc_mutex_lock_it.l_pid = 0; /* pid not actually interesting */ proc_mutex_unlock_it.l_whence = SEEK_SET; /* from current point */ proc_mutex_unlock_it.l_start = 0; /* -"- */ proc_mutex_unlock_it.l_len = 0; /* until end of file */ proc_mutex_unlock_it.l_type = F_UNLCK; /* set exclusive/write lock */ proc_mutex_unlock_it.l_pid = 0; /* pid not actually interesting */ } static apr_status_t proc_mutex_fcntl_cleanup(void *mutex_) { apr_status_t status = APR_SUCCESS; apr_proc_mutex_t *mutex=mutex_; if (mutex->curr_locked == 1) { status = proc_mutex_fcntl_release(mutex); if (status != APR_SUCCESS) return status; } if (mutex->interproc) { status = apr_file_close(mutex->interproc); } if (!mutex->interproc_closing && mutex->os.crossproc != -1 && close(mutex->os.crossproc) == -1 && status == APR_SUCCESS) { status = errno; } return status; } static apr_status_t proc_mutex_fcntl_create(apr_proc_mutex_t *new_mutex, const char *fname) { int rv; if (fname) { new_mutex->fname = apr_pstrdup(new_mutex->pool, fname); rv = apr_file_open(&new_mutex->interproc, new_mutex->fname, APR_FOPEN_CREATE | APR_FOPEN_WRITE | APR_FOPEN_EXCL, APR_UREAD | APR_UWRITE | APR_GREAD | APR_WREAD, new_mutex->pool); } else { new_mutex->fname = apr_pstrdup(new_mutex->pool, "/tmp/aprXXXXXX"); rv = apr_file_mktemp(&new_mutex->interproc, new_mutex->fname, APR_FOPEN_CREATE | APR_FOPEN_WRITE | APR_FOPEN_EXCL, new_mutex->pool); } if (rv != APR_SUCCESS) { return rv; } new_mutex->os.crossproc = new_mutex->interproc->filedes; new_mutex->interproc_closing = 1; new_mutex->curr_locked = 0; unlink(new_mutex->fname); apr_pool_cleanup_register(new_mutex->pool, (void*)new_mutex, apr_proc_mutex_cleanup, apr_pool_cleanup_null); return APR_SUCCESS; } static apr_status_t proc_mutex_fcntl_acquire(apr_proc_mutex_t *mutex) { int rc; do { rc = fcntl(mutex->os.crossproc, F_SETLKW, &proc_mutex_lock_it); } while (rc < 0 && errno == EINTR); if (rc < 0) { return errno; } mutex->curr_locked=1; return APR_SUCCESS; } static apr_status_t proc_mutex_fcntl_tryacquire(apr_proc_mutex_t *mutex) { int rc; do { rc = fcntl(mutex->os.crossproc, F_SETLK, &proc_mutex_lock_it); } while (rc < 0 && errno == EINTR); if (rc < 0) { #if FCNTL_TRYACQUIRE_EACCES if (errno == EACCES) { #else if (errno == EAGAIN) { #endif return APR_EBUSY; } return errno; } mutex->curr_locked = 1; return APR_SUCCESS; } static apr_status_t proc_mutex_fcntl_release(apr_proc_mutex_t *mutex) { int rc; mutex->curr_locked=0; do { rc = fcntl(mutex->os.crossproc, F_SETLKW, &proc_mutex_unlock_it); } while (rc < 0 && errno == EINTR); if (rc < 0) { return errno; } return APR_SUCCESS; } static apr_status_t proc_mutex_fcntl_perms_set(apr_proc_mutex_t *mutex, apr_fileperms_t perms, apr_uid_t uid, apr_gid_t gid) { if (mutex->fname) { if (!(perms & APR_FPROT_GSETID)) gid = -1; if (fchown(mutex->os.crossproc, uid, gid) < 0) { return errno; } } return APR_SUCCESS; } static const apr_proc_mutex_unix_lock_methods_t mutex_fcntl_methods = { #if APR_PROCESS_LOCK_IS_GLOBAL || !APR_HAS_THREADS || defined(FCNTL_IS_GLOBAL) APR_PROCESS_LOCK_MECH_IS_GLOBAL, #else 0, #endif proc_mutex_fcntl_create, proc_mutex_fcntl_acquire, proc_mutex_fcntl_tryacquire, proc_mutex_spinsleep_timedacquire, proc_mutex_fcntl_release, proc_mutex_fcntl_cleanup, proc_mutex_no_child_init, proc_mutex_fcntl_perms_set, APR_LOCK_FCNTL, "fcntl" }; #endif /* fcntl implementation */ #if APR_HAS_FLOCK_SERIALIZE static apr_status_t proc_mutex_flock_release(apr_proc_mutex_t *); static apr_status_t proc_mutex_flock_cleanup(void *mutex_) { apr_status_t status = APR_SUCCESS; apr_proc_mutex_t *mutex=mutex_; if (mutex->curr_locked == 1) { status = proc_mutex_flock_release(mutex); if (status != APR_SUCCESS) return status; } if (mutex->interproc) { /* if it was opened properly */ status = apr_file_close(mutex->interproc); } if (!mutex->interproc_closing && mutex->os.crossproc != -1 && close(mutex->os.crossproc) == -1 && status == APR_SUCCESS) { status = errno; } if (mutex->fname) { unlink(mutex->fname); } return status; } static apr_status_t proc_mutex_flock_create(apr_proc_mutex_t *new_mutex, const char *fname) { int rv; if (fname) { new_mutex->fname = apr_pstrdup(new_mutex->pool, fname); rv = apr_file_open(&new_mutex->interproc, new_mutex->fname, APR_FOPEN_CREATE | APR_FOPEN_WRITE | APR_FOPEN_EXCL, APR_UREAD | APR_UWRITE, new_mutex->pool); } else { new_mutex->fname = apr_pstrdup(new_mutex->pool, "/tmp/aprXXXXXX"); rv = apr_file_mktemp(&new_mutex->interproc, new_mutex->fname, APR_FOPEN_CREATE | APR_FOPEN_WRITE | APR_FOPEN_EXCL, new_mutex->pool); } if (rv != APR_SUCCESS) { proc_mutex_flock_cleanup(new_mutex); return rv; } new_mutex->os.crossproc = new_mutex->interproc->filedes; new_mutex->interproc_closing = 1; new_mutex->curr_locked = 0; apr_pool_cleanup_register(new_mutex->pool, (void *)new_mutex, apr_proc_mutex_cleanup, apr_pool_cleanup_null); return APR_SUCCESS; } static apr_status_t proc_mutex_flock_acquire(apr_proc_mutex_t *mutex) { int rc; do { rc = flock(mutex->os.crossproc, LOCK_EX); } while (rc < 0 && errno == EINTR); if (rc < 0) { return errno; } mutex->curr_locked = 1; return APR_SUCCESS; } static apr_status_t proc_mutex_flock_tryacquire(apr_proc_mutex_t *mutex) { int rc; do { rc = flock(mutex->os.crossproc, LOCK_EX | LOCK_NB); } while (rc < 0 && errno == EINTR); if (rc < 0) { if (errno == EWOULDBLOCK || errno == EAGAIN) { return APR_EBUSY; } return errno; } mutex->curr_locked = 1; return APR_SUCCESS; } static apr_status_t proc_mutex_flock_release(apr_proc_mutex_t *mutex) { int rc; mutex->curr_locked = 0; do { rc = flock(mutex->os.crossproc, LOCK_UN); } while (rc < 0 && errno == EINTR); if (rc < 0) { return errno; } return APR_SUCCESS; } static apr_status_t proc_mutex_flock_child_init(apr_proc_mutex_t **mutex, apr_pool_t *pool, const char *fname) { apr_proc_mutex_t *new_mutex; int rv; if (!fname) { fname = (*mutex)->fname; if (!fname) { return APR_SUCCESS; } } new_mutex = (apr_proc_mutex_t *)apr_pmemdup(pool, *mutex, sizeof(apr_proc_mutex_t)); new_mutex->pool = pool; new_mutex->fname = apr_pstrdup(pool, fname); rv = apr_file_open(&new_mutex->interproc, new_mutex->fname, APR_FOPEN_WRITE, 0, new_mutex->pool); if (rv != APR_SUCCESS) { return rv; } new_mutex->os.crossproc = new_mutex->interproc->filedes; new_mutex->interproc_closing = 1; *mutex = new_mutex; return APR_SUCCESS; } static apr_status_t proc_mutex_flock_perms_set(apr_proc_mutex_t *mutex, apr_fileperms_t perms, apr_uid_t uid, apr_gid_t gid) { if (mutex->fname) { if (!(perms & APR_FPROT_GSETID)) gid = -1; if (fchown(mutex->os.crossproc, uid, gid) < 0) { return errno; } } return APR_SUCCESS; } static const apr_proc_mutex_unix_lock_methods_t mutex_flock_methods = { #if APR_PROCESS_LOCK_IS_GLOBAL || !APR_HAS_THREADS || defined(FLOCK_IS_GLOBAL) APR_PROCESS_LOCK_MECH_IS_GLOBAL, #else 0, #endif proc_mutex_flock_create, proc_mutex_flock_acquire, proc_mutex_flock_tryacquire, proc_mutex_spinsleep_timedacquire, proc_mutex_flock_release, proc_mutex_flock_cleanup, proc_mutex_flock_child_init, proc_mutex_flock_perms_set, APR_LOCK_FLOCK, "flock" }; #endif /* flock implementation */ void apr_proc_mutex_unix_setup_lock(void) { /* setup only needed for sysvsem and fnctl */ #if APR_HAS_SYSVSEM_SERIALIZE proc_mutex_sysv_setup(); #endif #if APR_HAS_FCNTL_SERIALIZE proc_mutex_fcntl_setup(); #endif } static apr_status_t proc_mutex_choose_method(apr_proc_mutex_t *new_mutex, apr_lockmech_e mech, apr_os_proc_mutex_t *ospmutex) { #if APR_HAS_PROC_PTHREAD_SERIALIZE new_mutex->os.pthread_interproc = NULL; #endif #if APR_HAS_POSIXSEM_SERIALIZE new_mutex->os.psem_interproc = NULL; #endif #if APR_HAS_SYSVSEM_SERIALIZE || APR_HAS_FCNTL_SERIALIZE || APR_HAS_FLOCK_SERIALIZE new_mutex->os.crossproc = -1; #if APR_HAS_FCNTL_SERIALIZE || APR_HAS_FLOCK_SERIALIZE new_mutex->interproc = NULL; new_mutex->interproc_closing = 0; #endif #endif switch (mech) { case APR_LOCK_FCNTL: #if APR_HAS_FCNTL_SERIALIZE new_mutex->meth = &mutex_fcntl_methods; if (ospmutex) { if (ospmutex->crossproc == -1) { return APR_EINVAL; } new_mutex->os.crossproc = ospmutex->crossproc; } #else return APR_ENOTIMPL; #endif break; case APR_LOCK_FLOCK: #if APR_HAS_FLOCK_SERIALIZE new_mutex->meth = &mutex_flock_methods; if (ospmutex) { if (ospmutex->crossproc == -1) { return APR_EINVAL; } new_mutex->os.crossproc = ospmutex->crossproc; } #else return APR_ENOTIMPL; #endif break; case APR_LOCK_SYSVSEM: #if APR_HAS_SYSVSEM_SERIALIZE new_mutex->meth = &mutex_sysv_methods; if (ospmutex) { if (ospmutex->crossproc == -1) { return APR_EINVAL; } new_mutex->os.crossproc = ospmutex->crossproc; } #else return APR_ENOTIMPL; #endif break; case APR_LOCK_POSIXSEM: #if APR_HAS_POSIXSEM_SERIALIZE new_mutex->meth = &mutex_posixsem_methods; if (ospmutex) { if (ospmutex->psem_interproc == NULL) { return APR_EINVAL; } new_mutex->os.psem_interproc = ospmutex->psem_interproc; } #else return APR_ENOTIMPL; #endif break; case APR_LOCK_PROC_PTHREAD: #if APR_HAS_PROC_PTHREAD_SERIALIZE new_mutex->meth = &mutex_proc_pthread_methods; if (ospmutex) { if (ospmutex->pthread_interproc == NULL) { return APR_EINVAL; } new_mutex->os.pthread_interproc = ospmutex->pthread_interproc; } #else return APR_ENOTIMPL; #endif break; case APR_LOCK_DEFAULT_TIMED: #if APR_HAS_PROC_PTHREAD_SERIALIZE \ && (APR_USE_PROC_PTHREAD_MUTEX_COND \ || defined(HAVE_PTHREAD_MUTEX_TIMEDLOCK)) \ && defined(HAVE_PTHREAD_MUTEX_ROBUST) #if APR_USE_PROC_PTHREAD_MUTEX_COND new_mutex->meth = &mutex_proc_pthread_cond_methods; #else new_mutex->meth = &mutex_proc_pthread_methods; #endif if (ospmutex) { if (ospmutex->pthread_interproc == NULL) { return APR_EINVAL; } new_mutex->os.pthread_interproc = ospmutex->pthread_interproc; } break; #elif APR_HAS_SYSVSEM_SERIALIZE && defined(HAVE_SEMTIMEDOP) new_mutex->meth = &mutex_sysv_methods; if (ospmutex) { if (ospmutex->crossproc == -1) { return APR_EINVAL; } new_mutex->os.crossproc = ospmutex->crossproc; } break; #elif APR_HAS_POSIXSEM_SERIALIZE && defined(HAVE_SEM_TIMEDWAIT) new_mutex->meth = &mutex_posixsem_methods; if (ospmutex) { if (ospmutex->psem_interproc == NULL) { return APR_EINVAL; } new_mutex->os.psem_interproc = ospmutex->psem_interproc; } break; #endif /* fall trough */ case APR_LOCK_DEFAULT: #if APR_USE_FLOCK_SERIALIZE new_mutex->meth = &mutex_flock_methods; if (ospmutex) { if (ospmutex->crossproc == -1) { return APR_EINVAL; } new_mutex->os.crossproc = ospmutex->crossproc; } #elif APR_USE_SYSVSEM_SERIALIZE new_mutex->meth = &mutex_sysv_methods; if (ospmutex) { if (ospmutex->crossproc == -1) { return APR_EINVAL; } new_mutex->os.crossproc = ospmutex->crossproc; } #elif APR_USE_FCNTL_SERIALIZE new_mutex->meth = &mutex_fcntl_methods; if (ospmutex) { if (ospmutex->crossproc == -1) { return APR_EINVAL; } new_mutex->os.crossproc = ospmutex->crossproc; } #elif APR_USE_PROC_PTHREAD_SERIALIZE new_mutex->meth = &mutex_proc_pthread_methods; if (ospmutex) { if (ospmutex->pthread_interproc == NULL) { return APR_EINVAL; } new_mutex->os.pthread_interproc = ospmutex->pthread_interproc; } #elif APR_USE_POSIXSEM_SERIALIZE new_mutex->meth = &mutex_posixsem_methods; if (ospmutex) { if (ospmutex->psem_interproc == NULL) { return APR_EINVAL; } new_mutex->os.psem_interproc = ospmutex->psem_interproc; } #else return APR_ENOTIMPL; #endif break; default: return APR_ENOTIMPL; } return APR_SUCCESS; } APR_DECLARE(const char *) apr_proc_mutex_defname(void) { apr_status_t rv; apr_proc_mutex_t mutex; if ((rv = proc_mutex_choose_method(&mutex, APR_LOCK_DEFAULT, NULL)) != APR_SUCCESS) { return "unknown"; } return apr_proc_mutex_name(&mutex); } static apr_status_t proc_mutex_create(apr_proc_mutex_t *new_mutex, apr_lockmech_e mech, const char *fname) { apr_status_t rv; if ((rv = proc_mutex_choose_method(new_mutex, mech, NULL)) != APR_SUCCESS) { return rv; } if ((rv = new_mutex->meth->create(new_mutex, fname)) != APR_SUCCESS) { return rv; } return APR_SUCCESS; } APR_DECLARE(apr_status_t) apr_proc_mutex_create(apr_proc_mutex_t **mutex, const char *fname, apr_lockmech_e mech, apr_pool_t *pool) { apr_proc_mutex_t *new_mutex; apr_status_t rv; new_mutex = apr_pcalloc(pool, sizeof(apr_proc_mutex_t)); new_mutex->pool = pool; if ((rv = proc_mutex_create(new_mutex, mech, fname)) != APR_SUCCESS) return rv; *mutex = new_mutex; return APR_SUCCESS; } APR_DECLARE(apr_status_t) apr_proc_mutex_child_init(apr_proc_mutex_t **mutex, const char *fname, apr_pool_t *pool) { return (*mutex)->meth->child_init(mutex, pool, fname); } APR_DECLARE(apr_status_t) apr_proc_mutex_lock(apr_proc_mutex_t *mutex) { return mutex->meth->acquire(mutex); } APR_DECLARE(apr_status_t) apr_proc_mutex_trylock(apr_proc_mutex_t *mutex) { return mutex->meth->tryacquire(mutex); } APR_DECLARE(apr_status_t) apr_proc_mutex_timedlock(apr_proc_mutex_t *mutex, apr_interval_time_t timeout) { #if APR_HAS_TIMEDLOCKS return mutex->meth->timedacquire(mutex, timeout); #else return APR_ENOTIMPL; #endif } APR_DECLARE(apr_status_t) apr_proc_mutex_unlock(apr_proc_mutex_t *mutex) { return mutex->meth->release(mutex); } APR_DECLARE(apr_status_t) apr_proc_mutex_cleanup(void *mutex) { return ((apr_proc_mutex_t *)mutex)->meth->cleanup(mutex); } APR_DECLARE(apr_lockmech_e) apr_proc_mutex_mech(apr_proc_mutex_t *mutex) { return mutex->meth->mech; } APR_DECLARE(const char *) apr_proc_mutex_name(apr_proc_mutex_t *mutex) { return mutex->meth->name; } APR_DECLARE(const char *) apr_proc_mutex_lockfile(apr_proc_mutex_t *mutex) { /* POSIX sems use the fname field but don't use a file, * so be careful. */ #if APR_HAS_FLOCK_SERIALIZE if (mutex->meth == &mutex_flock_methods) { return mutex->fname; } #endif #if APR_HAS_FCNTL_SERIALIZE if (mutex->meth == &mutex_fcntl_methods) { return mutex->fname; } #endif return NULL; } APR_PERMS_SET_IMPLEMENT(proc_mutex) { apr_proc_mutex_t *mutex = (apr_proc_mutex_t *)theproc_mutex; return mutex->meth->perms_set(mutex, perms, uid, gid); } APR_POOL_IMPLEMENT_ACCESSOR(proc_mutex) /* Implement OS-specific accessors defined in apr_portable.h */ APR_DECLARE(apr_status_t) apr_os_proc_mutex_get_ex(apr_os_proc_mutex_t *ospmutex, apr_proc_mutex_t *pmutex, apr_lockmech_e *mech) { *ospmutex = pmutex->os; if (mech) { *mech = pmutex->meth->mech; } return APR_SUCCESS; } APR_DECLARE(apr_status_t) apr_os_proc_mutex_get(apr_os_proc_mutex_t *ospmutex, apr_proc_mutex_t *pmutex) { return apr_os_proc_mutex_get_ex(ospmutex, pmutex, NULL); } APR_DECLARE(apr_status_t) apr_os_proc_mutex_put_ex(apr_proc_mutex_t **pmutex, apr_os_proc_mutex_t *ospmutex, apr_lockmech_e mech, int register_cleanup, apr_pool_t *pool) { apr_status_t rv; if (pool == NULL) { return APR_ENOPOOL; } if ((*pmutex) == NULL) { (*pmutex) = (apr_proc_mutex_t *)apr_pcalloc(pool, sizeof(apr_proc_mutex_t)); (*pmutex)->pool = pool; } rv = proc_mutex_choose_method(*pmutex, mech, ospmutex); #if APR_HAS_FCNTL_SERIALIZE || APR_HAS_FLOCK_SERIALIZE if (rv == APR_SUCCESS) { rv = apr_os_file_put(&(*pmutex)->interproc, &(*pmutex)->os.crossproc, 0, pool); } #endif if (rv == APR_SUCCESS && register_cleanup) { apr_pool_cleanup_register(pool, *pmutex, apr_proc_mutex_cleanup, apr_pool_cleanup_null); } return rv; } APR_DECLARE(apr_status_t) apr_os_proc_mutex_put(apr_proc_mutex_t **pmutex, apr_os_proc_mutex_t *ospmutex, apr_pool_t *pool) { return apr_os_proc_mutex_put_ex(pmutex, ospmutex, APR_LOCK_DEFAULT, 0, pool); }