123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562 |
- /*
- * Copyright (c) 2006-2024 RT-Thread Development Team
- *
- * SPDX-License-Identifier: Apache-2.0
- *
- * Change Logs:
- * Date Author Notes
- * 2010-10-26 Bernard the first version
- */
- #include <pthread.h>
- int pthread_rwlockattr_init(pthread_rwlockattr_t *attr)
- {
- if (!attr)
- return EINVAL;
- *attr = PTHREAD_PROCESS_PRIVATE;
- return 0;
- }
- RTM_EXPORT(pthread_rwlockattr_init);
- int pthread_rwlockattr_destroy(pthread_rwlockattr_t *attr)
- {
- if (!attr)
- return EINVAL;
- return 0;
- }
- RTM_EXPORT(pthread_rwlockattr_destroy);
- int pthread_rwlockattr_getpshared(const pthread_rwlockattr_t *attr,
- int *pshared)
- {
- if (!attr || !pshared)
- return EINVAL;
- *pshared = PTHREAD_PROCESS_PRIVATE;
- return 0;
- }
- RTM_EXPORT(pthread_rwlockattr_getpshared);
- int pthread_rwlockattr_setpshared(pthread_rwlockattr_t *attr, int pshared)
- {
- if (!attr || pshared != PTHREAD_PROCESS_PRIVATE)
- return EINVAL;
- return 0;
- }
- RTM_EXPORT(pthread_rwlockattr_setpshared);
- /**
- * @brief Initializes a read-write lock.
- *
- * This function initializes the read-write lock object pointed to by `rwlock` with the
- * attributes specified by `attr`. If `attr` is `NULL`, the default attributes are used.
- * A read-write lock allows multiple threads to read or a single thread to write, but not both simultaneously.
- *
- * @param rwlock A pointer to the read-write lock object to be initialized.
- * Must point to valid memory.
- * @param attr A pointer to the attributes for the read-write lock.
- * If `NULL`, default attributes are applied.
- *
- * @return
- * - `0` on success.
- * - A non-zero error code on failure, including:
- * - `EINVAL`: Invalid attributes.
- *
- * @note
- * - The read-write lock must be destroyed using `pthread_rwlock_destroy()` when it is no longer needed.
- * - 'rw_mutex' is used for protecting rwlock data.
- * 'rw_condreaders' is a condition variable for controlling readers.
- * 'rw_condwriters' is a condition variable for controlling writers.
- *
- * @see pthread_rwlock_destroy, pthread_rwlock_rdlock, pthread_rwlock_wrlock, pthread_rwlock_unlock
- */
- int pthread_rwlock_init(pthread_rwlock_t *rwlock,
- const pthread_rwlockattr_t *attr)
- {
- if (!rwlock)
- return EINVAL;
- rwlock->attr = PTHREAD_PROCESS_PRIVATE;
- pthread_mutex_init(&(rwlock->rw_mutex), NULL);
- pthread_cond_init(&(rwlock->rw_condreaders), NULL);
- pthread_cond_init(&(rwlock->rw_condwriters), NULL);
- rwlock->rw_nwaitwriters = 0;
- rwlock->rw_nwaitreaders = 0;
- rwlock->rw_refcount = 0;
- return 0;
- }
- RTM_EXPORT(pthread_rwlock_init);
- /**
- * @brief Destroys a read-write lock.
- *
- * This function destroys the read-write lock object pointed to by `rwlock`. After
- * the lock is destroyed, it cannot be used until it is reinitialized with
- * `pthread_rwlock_init`. Any threads currently blocked on the lock are affected by the destruction.
- *
- * @param rwlock A pointer to the read-write lock object to be destroyed.
- * Must point to a valid, initialized read-write lock.
- *
- * @return
- * - `0` on success.
- * - A non-zero error code on failure, including:
- * - `EINVAL`: The `rwlock` is invalid or uninitialized.
- * - `EBUSY`: The lock is currently in use by a thread, and cannot be destroyed.
- *
- * @note
- * - The read-write lock must not be in use (i.e., no threads should be blocked on it)
- * when `pthread_rwlock_destroy` is called.
- * - Calling this function on an uninitialized or destroyed lock will result in undefined behavior.
- * - Ensure that all threads have unlocked the lock before attempting to destroy it.
- *
- * @see pthread_rwlock_init, pthread_rwlock_rdlock, pthread_rwlock_wrlock, pthread_rwlock_unlock
- */
- int pthread_rwlock_destroy (pthread_rwlock_t *rwlock)
- {
- int result;
- if (!rwlock)
- return EINVAL;
- if (rwlock->attr == -1)
- return 0; /* rwlock is not initialized */
- if ( (result = pthread_mutex_lock(&rwlock->rw_mutex)) != 0)
- return(result);
- if (rwlock->rw_refcount != 0 ||
- rwlock->rw_nwaitreaders != 0 ||
- rwlock->rw_nwaitwriters != 0)
- {
- result = EBUSY;
- return result;
- }
- else
- {
- /* check whether busy */
- result = rt_sem_trytake(&(rwlock->rw_condreaders.sem));
- if (result == RT_EOK)
- {
- result = rt_sem_trytake(&(rwlock->rw_condwriters.sem));
- if (result == RT_EOK)
- {
- rt_sem_release(&(rwlock->rw_condreaders.sem));
- rt_sem_release(&(rwlock->rw_condwriters.sem));
- pthread_cond_destroy(&rwlock->rw_condreaders);
- pthread_cond_destroy(&rwlock->rw_condwriters);
- }
- else
- {
- rt_sem_release(&(rwlock->rw_condreaders.sem));
- result = EBUSY;
- }
- }
- else
- result = EBUSY;
- }
- pthread_mutex_unlock(&rwlock->rw_mutex);
- if (result == 0)
- pthread_mutex_destroy(&rwlock->rw_mutex);
- return result;
- }
- RTM_EXPORT(pthread_rwlock_destroy);
- /**
- * @brief Acquire a read lock on a read-write lock.
- *
- * This function locks the specified read-write lock for reading. If the lock
- * is already held by one or more threads for reading, the calling thread
- * can acquire the lock as well (shared access). However, if the lock is
- * held by another writer thread, or other writer thread has been waiting
- * for the lock, the calling thread will block until the write lock is released.
- *
- * @param rwlock A pointer to the read-write lock to be locked.
- *
- * @return - 0 on success.
- * - EINVAL if the rwlock is invalid.
- * - EDEADLK if a deadlock condition is detected (optional; implementation-dependent).
- *
- * @note A thread that has acquired a read lock must eventually release it
- * using `pthread_rwlock_unlock`. Multiple read locks can be held
- * simultaneously, but a write lock excludes all other locks.
- *
- * @see pthread_rwlock_unlock
- * @see pthread_rwlock_wrlock
- * @see pthread_rwlock_tryrdlock
- */
- int pthread_rwlock_rdlock(pthread_rwlock_t *rwlock)
- {
- int result;
- if (!rwlock)
- return EINVAL;
- if (rwlock->attr == -1)
- pthread_rwlock_init(rwlock, NULL);
- if ((result = pthread_mutex_lock(&rwlock->rw_mutex)) != 0)
- return(result);
- /* give preference to waiting writers */
- while (rwlock->rw_refcount < 0 || rwlock->rw_nwaitwriters > 0)
- {
- rwlock->rw_nwaitreaders++;
- /* rw_mutex will be released when waiting for rw_condreaders */
- result = pthread_cond_wait(&rwlock->rw_condreaders, &rwlock->rw_mutex);
- /* rw_mutex should have been taken again when returned from waiting */
- rwlock->rw_nwaitreaders--;
- if (result != 0) /* wait error */
- break;
- }
- /* another reader has a read lock */
- if (result == 0)
- rwlock->rw_refcount++;
- pthread_mutex_unlock(&rwlock->rw_mutex);
- return (result);
- }
- RTM_EXPORT(pthread_rwlock_rdlock);
- /**
- * @brief Try to acquire a read lock on a read-write lock without blocking.
- *
- * This function attempts to acquire a read lock on the specified read-write lock.
- * If the lock is already held for writing, the function will return immediately
- * without blocking the calling thread. If the lock is available for reading,
- * it will be acquired successfully.
- *
- * @param rwlock A pointer to the read-write lock to attempt to lock.
- *
- * @return - 0 on success, indicating the read lock was acquired.
- * - EBUSY if the lock is currently held for writing by another thread.
- * - EINVAL if the rwlock is invalid.
- *
- * @note This function is non-blocking and returns immediately if the lock
- * cannot be acquired. After successfully acquiring the read lock,
- * the thread must release it using `pthread_rwlock_unlock`.
- *
- * @see pthread_rwlock_unlock
- * @see pthread_rwlock_rdlock
- * @see pthread_rwlock_trywrlock
- */
- int pthread_rwlock_tryrdlock(pthread_rwlock_t *rwlock)
- {
- int result;
- if (!rwlock)
- return EINVAL;
- if (rwlock->attr == -1)
- pthread_rwlock_init(rwlock, NULL);
- if ((result = pthread_mutex_lock(&rwlock->rw_mutex)) != 0)
- return(result);
- if (rwlock->rw_refcount < 0 || rwlock->rw_nwaitwriters > 0)
- result = EBUSY; /* held by a writer or waiting writers */
- else
- rwlock->rw_refcount++; /* increment count of reader locks */
- pthread_mutex_unlock(&rwlock->rw_mutex);
- return(result);
- }
- RTM_EXPORT(pthread_rwlock_tryrdlock);
- /**
- * @brief Acquire a read lock on a read-write lock with a timeout.
- *
- * This function attempts to lock the specified read-write lock for reading,
- * blocking until the lock becomes available or the specified timeout expires.
- * If the lock is held for writing by another thread, the calling thread will
- * block, but only up to the time specified by `abstime`.
- *
- * @param rwlock A pointer to the read-write lock to be locked.
- * @param abstime A pointer to a `timespec` structure specifying the
- * absolute timeout (in seconds and nanoseconds since the
- * Epoch, 1970-01-01 00:00:00 UTC).
- *
- * @return - 0 on success, indicating the read lock was acquired.
- * - ETIMEDOUT if the timeout expired before the lock could be acquired.
- * - EINVAL if the `rwlock` is invalid or `abstime` contains invalid values.
- * - EDEADLK if a deadlock condition is detected (optional; implementation-dependent).
- *
- * @note The timeout is specified as an absolute time (not relative). After
- * acquiring the read lock, the thread must release it using
- * `pthread_rwlock_unlock`.
- *
- * @warning If the system clock is changed (e.g., via manual adjustment or
- * NTP synchronization), the timeout behavior may be affected.
- *
- * @see pthread_rwlock_unlock
- * @see pthread_rwlock_rdlock
- * @see pthread_rwlock_tryrdlock
- */
- int pthread_rwlock_timedrdlock(pthread_rwlock_t *rwlock,
- const struct timespec *abstime)
- {
- int result;
- if (!rwlock)
- return EINVAL;
- if (rwlock->attr == -1)
- pthread_rwlock_init(rwlock, NULL);
- if ( (result = pthread_mutex_lock(&rwlock->rw_mutex)) != 0)
- return(result);
- /* give preference to waiting writers */
- while (rwlock->rw_refcount < 0 || rwlock->rw_nwaitwriters > 0)
- {
- rwlock->rw_nwaitreaders++;
- /* rw_mutex will be released when waiting for rw_condreaders */
- result = pthread_cond_timedwait(&rwlock->rw_condreaders, &rwlock->rw_mutex, abstime);
- /* rw_mutex should have been taken again when returned from waiting */
- rwlock->rw_nwaitreaders--;
- if (result != 0)
- break;
- }
- /* another reader has a read lock */
- if (result == 0)
- rwlock->rw_refcount++;
- pthread_mutex_unlock(&rwlock->rw_mutex);
- return (result);
- }
- RTM_EXPORT(pthread_rwlock_timedrdlock);
- /**
- * @brief Acquire a write lock on a read-write lock with a timeout.
- *
- * This function attempts to acquire a write lock on the specified read-write lock,
- * blocking until the lock becomes available or the specified timeout expires.
- * If the lock is already held for reading or writing by another thread, the
- * calling thread will block, but only up to the time specified by `abstime`.
- *
- * @param rwlock A pointer to the read-write lock to be locked.
- * @param abstime A pointer to a `timespec` structure specifying the
- * absolute timeout (in seconds and nanoseconds since the
- * Epoch, 1970-01-01 00:00:00 UTC).
- *
- * @return
- * - 0 on success, indicating the write lock was acquired.
- * - EINVAL if the `rwlock` is invalid or `abstime` contains invalid values.
- * - EDEADLK if a deadlock condition is detected (optional; implementation-dependent).
- *
- * @note The timeout is specified as an absolute time (not relative). After
- * acquiring the write lock, the thread must release it using
- * `pthread_rwlock_unlock`.
- *
- * @warning If the system clock is adjusted (e.g., manually or via NTP synchronization),
- * the timeout behavior may be affected.
- *
- * @see pthread_rwlock_unlock
- * @see pthread_rwlock_wrlock
- * @see pthread_rwlock_trywrlock
- */
- int pthread_rwlock_timedwrlock(pthread_rwlock_t *rwlock,
- const struct timespec *abstime)
- {
- int result;
- if (!rwlock)
- return EINVAL;
- if (rwlock->attr == -1)
- pthread_rwlock_init(rwlock, NULL);
- if ((result = pthread_mutex_lock(&rwlock->rw_mutex)) != 0)
- return(result);
- while (rwlock->rw_refcount != 0)
- {
- rwlock->rw_nwaitwriters++;
- /* rw_mutex will be released when waiting for rw_condwriters */
- result = pthread_cond_timedwait(&rwlock->rw_condwriters, &rwlock->rw_mutex, abstime);
- /* rw_mutex should have been taken again when returned from waiting */
- rwlock->rw_nwaitwriters--;
- if (result != 0)
- break;
- }
- if (result == 0)
- rwlock->rw_refcount = -1;
- pthread_mutex_unlock(&rwlock->rw_mutex);
- return(result);
- }
- RTM_EXPORT(pthread_rwlock_timedwrlock);
- /**
- * @brief Try to acquire a write lock on a read-write lock without blocking.
- *
- * This function attempts to acquire a write lock on the specified read-write lock.
- * If the lock is already held for reading or writing by another thread, the function
- * will return immediately without blocking the calling thread. If the lock is
- * available, it will be acquired successfully.
- *
- * @param rwlock A pointer to the read-write lock to attempt to lock.
- *
- * @return
- * - 0 on success, indicating the write lock was acquired.
- * - EBUSY if the lock is currently held by another thread (read or write lock).
- * - EINVAL if the `rwlock` is invalid.
- *
- * @note This function is non-blocking and returns immediately if the lock cannot
- * be acquired. After successfully acquiring the write lock, the thread must
- * release it using `pthread_rwlock_unlock`.
- *
- * @see pthread_rwlock_unlock
- * @see pthread_rwlock_wrlock
- * @see pthread_rwlock_timedwrlock
- */
- int pthread_rwlock_trywrlock(pthread_rwlock_t *rwlock)
- {
- int result;
- if (!rwlock)
- return EINVAL;
- if (rwlock->attr == -1)
- pthread_rwlock_init(rwlock, NULL);
- if ((result = pthread_mutex_lock(&rwlock->rw_mutex)) != 0)
- return(result);
- if (rwlock->rw_refcount != 0)
- result = EBUSY; /* held by either writer or reader(s) */
- else
- rwlock->rw_refcount = -1; /* available, indicate a writer has it */
- pthread_mutex_unlock(&rwlock->rw_mutex);
- return(result);
- }
- RTM_EXPORT(pthread_rwlock_trywrlock);
- /**
- * @brief Release a read or write lock on a read-write lock.
- *
- * This function unlocks the specified read-write lock, releasing either a read
- * lock or a write lock held by the calling thread. If the calling thread does
- * not hold the lock, the behavior is undefined.
- *
- * @param rwlock A pointer to the read-write lock to be unlocked.
- *
- * @return
- * - 0 on success.
- * - EINVAL if the `rwlock` is invalid.
- *
- * @note
- * - This function must only be called by the thread that successfully acquired
- * the lock.
- *
- * @see pthread_rwlock_rdlock
- * @see pthread_rwlock_wrlock
- * @see pthread_rwlock_tryrdlock
- * @see pthread_rwlock_trywrlock
- */
- int pthread_rwlock_unlock(pthread_rwlock_t *rwlock)
- {
- int result;
- if (!rwlock)
- return EINVAL;
- if (rwlock->attr == -1)
- pthread_rwlock_init(rwlock, NULL);
- if ( (result = pthread_mutex_lock(&rwlock->rw_mutex)) != 0)
- return(result);
- if (rwlock->rw_refcount > 0)
- rwlock->rw_refcount--; /* releasing a reader */
- else if (rwlock->rw_refcount == -1)
- rwlock->rw_refcount = 0; /* releasing a writer */
- /* give preference to waiting writers over waiting readers */
- if (rwlock->rw_nwaitwriters > 0)
- {
- if (rwlock->rw_refcount == 0)
- result = pthread_cond_signal(&rwlock->rw_condwriters);
- }
- else if (rwlock->rw_nwaitreaders > 0)
- {
- result = pthread_cond_broadcast(&rwlock->rw_condreaders);
- }
- pthread_mutex_unlock(&rwlock->rw_mutex);
- return(result);
- }
- RTM_EXPORT(pthread_rwlock_unlock);
- /**
- * @brief Acquire a write lock on a read-write lock.
- *
- * This function locks the specified read-write lock for writing. If the lock
- * is already held by another thread for reading or writing, the calling thread
- * blocks until the lock becomes available.
- *
- * @param rwlock A pointer to the read-write lock to be locked.
- *
- * @return
- * - 0 on success, indicating the write lock was acquired.
- * - EINVAL if the `rwlock` is invalid.
- * - EDEADLK if a deadlock condition is detected (optional; implementation-dependent).
- *
- * @note
- * - A write lock is exclusive, meaning no other thread can acquire a read or
- * write lock while a write lock is held.
- * - The thread that successfully acquires the write lock must release it using
- * `pthread_rwlock_unlock`.
- *
- * @see pthread_rwlock_unlock
- * @see pthread_rwlock_trywrlock
- * @see pthread_rwlock_timedwrlock
- * @see pthread_rwlock_rdlock
- */
- int pthread_rwlock_wrlock(pthread_rwlock_t *rwlock)
- {
- int result;
- if (!rwlock)
- return EINVAL;
- if (rwlock->attr == -1)
- pthread_rwlock_init(rwlock, NULL);
- if ((result = pthread_mutex_lock(&rwlock->rw_mutex)) != 0)
- return(result);
- while (rwlock->rw_refcount != 0)
- {
- rwlock->rw_nwaitwriters++;
- /* rw_mutex will be released when waiting for rw_condwriters */
- result = pthread_cond_wait(&rwlock->rw_condwriters, &rwlock->rw_mutex);
- /* rw_mutex should have been taken again when returned from waiting */
- rwlock->rw_nwaitwriters--;
- if (result != 0)
- break;
- }
- if (result == 0)
- rwlock->rw_refcount = -1;
- pthread_mutex_unlock(&rwlock->rw_mutex);
- return(result);
- }
- RTM_EXPORT(pthread_rwlock_wrlock);
|