pthread_rwlock_wrlock
时间: 2023-04-19 10:00:09 浏览: 441
pthread_rwlock_wrlock是一个pthread库中的函数,用于在多线程环境下实现读写锁的写锁操作。当一个线程调用pthread_rwlock_wrlock函数时,它会尝试获取写锁,如果当前没有其他线程持有读锁或写锁,则该线程可以获取写锁并继续执行。如果有其他线程持有读锁或写锁,则该线程会被阻塞,直到它可以获取写锁为止。
相关问题
基于rt thread 的信号量实现读写锁的控制结构体和相关API函数,如调用pthread_rwlock_init()初始化一个读写锁,写线程调用pthread_rwlock_wrlock()对读写锁写锁定,读线程调用pthread_rwlock_rdlock()对读写锁读锁定,当不需要使用此读写锁时调用pthread_rwlock_destroy()销毁读写锁等,要求条理清晰
在基于 rt-thread 的系统中,可以使用 `struct rt_semaphore` 结构体来实现读写锁的控制。具体实现可以参考以下代码:
```
struct rt_semaphore rw_sem;
int read_count;
/* 初始化读写锁 */
int pthread_rwlock_init(pthread_rwlock_t *rwlock, const pthread_rwlockattr_t *attr)
{
/* 初始化信号量,初始值为 1 */
rt_sem_init(&rw_sem, "rw_sem", 1, RT_IPC_FLAG_FIFO);
read_count = 0;
return 0;
}
/* 销毁读写锁 */
int pthread_rwlock_destroy(pthread_rwlock_t *rwlock)
{
/* 删除信号量 */
rt_sem_delete(&rw_sem);
return 0;
}
/* 对读写锁进行读锁定 */
int pthread_rwlock_rdlock(pthread_rwlock_t *rwlock)
{
/* 获取信号量,如果信号量值为 0,则线程阻塞 */
rt_sem_take(&rw_sem, RT_WAITING_FOREVER);
read_count++;
return 0;
}
/* 对读写锁进行写锁定 */
int pthread_rwlock_wrlock(pthread_rwlock_t *rwlock)
{
/* 获取信号量,如果信号量值为 0,则线程阻塞 */
rt_sem_take(&rw_sem, RT_WAITING_FOREVER);
while (read_count > 0) {
/* 如果有读线程在使用,则等待读线程全部释放 */
rt_sem_release(&rw_sem);
rt_sem_take(&rw_sem, RT_WAITING_FOREVER);
}
return 0;
}
/* 对读写锁进行解锁 */
int pthread_rwlock_unlock(pthread_rwlock_t *rwlock)
{
/* 释放信号量 */
rt_sem_release(&rw_sem);
read_count--;
return 0;
}
```
在上述代码中,我们使用了一个信号量来实现读写锁的控制。信号量的初始值为 1,表示读写锁是未锁定状态。当线程需要对读写锁进行锁定时,首先需要获取信号量,如果信号量的值为 0,则线程阻塞等待。对于读锁,只需要将读计数器 `read_count` 加 1 即可;对于写锁,则需要判断是否有读线程在使用,如果有,则等待读线程全部释放后再获取锁。对于解锁操作,只需要释放信号量并将读计数器减 1 即可。
上述代码中的函数名和参数与 POSIX 标准中定义的函数名和参数保持一致,因此可以直接调用这些函数来实现读写锁的控制。
分析一下下面这段代码while(1) { revents = 0; #ifndef DISABLE_LIBSSH if (session->ssh_chan != NULL) { /* we are getting data from libssh's channel */ status = ssh_channel_poll_timeout(session->ssh_chan, timeout, 0); if (status > 0) { revents = POLLIN; } } else #endif #ifdef ENABLE_TLS if (session->tls != NULL) { /* we are getting data from TLS session using OpenSSL */ fds.fd = SSL_get_fd(session->tls); fds.events = POLLIN; fds.revents = 0; status = poll(&fds, 1, timeout); revents = (unsigned long int) fds.revents; } else #endif if (session->fd_input != -1) { /* we are getting data from standard file descriptor */ fds.fd = session->fd_input; fds.events = POLLIN; fds.revents = 0; status = poll(&fds, 1, timeout); revents = (unsigned long int) fds.revents; } else { ERROR("Invalid session to receive data."); return (NC_MSG_UNKNOWN); } /* process the result */ if (status == 0) { /* timed out */ DBG_UNLOCK("mut_channel"); pthread_mutex_unlock(session->mut_channel); return (NC_MSG_WOULDBLOCK); } else if (((status == -1) && (errno == EINTR)) #ifndef DISABLE_LIBSSH || (status == SSH_AGAIN) #endif ) { /* poll was interrupted */ continue; } else if (status < 0) { /* poll failed - something wrong happend, close this socket and wait for another request */ DBG_UNLOCK("mut_channel"); pthread_mutex_unlock(session->mut_channel); #ifndef DISABLE_LIBSSH if (status == SSH_EOF) { emsg = "end of file"; } else if (!session->ssh_chan) { emsg = strerror(errno); } else if (session->ssh_sess) { emsg = ssh_get_error(session->ssh_sess); } else { emsg = "description not available"; } #else emsg = strerror(errno); #endif WARN("Input channel error (%s)", emsg); nc_session_close(session, NC_SESSION_TERM_DROPPED); if (nc_info) { pthread_rwlock_wrlock(&(nc_info->lock)); nc_info->stats.sessions_dropped++; pthread_rwlock_unlock(&(nc_info->lock)); } return (NC_MSG_UNKNOWN); } /* status > 0 */ /* check the status of the socket */ /* if nothing to read and POLLHUP (EOF) or POLLERR set */ if ((revents & POLLHUP) || (revents & POLLERR)) { /* close client's socket (it's probably already closed by client */ DBG_UNLOCK("mut_channel"); pthread_mutex_unlock(session->mut_channel); ERROR("Input channel closed"); nc_session_close(session, NC_SESSION_TERM_DROPPED); if (nc_info) { pthread_rwlock_wrlock(&(nc_info->lock)); nc_info->stats.sessions_dropped++; pthread_rwlock_unlock(&(nc_info->lock)); } return (NC_MSG_UNKNOWN); } /* we have something to read */ break; }
这段代码是一个循环,用于从网络通道中读取数据,直到有数据可读或者等待超时。在循环中,代码会根据条件编译选择不同的代码路径,从不同的网络通道中获取数据。当获取到数据后,代码会根据获取到的数据的状态进行处理。如果等待超时,则返回NC_MSG_WOULDBLOCK表示没有数据可读;如果获取到的数据出现错误,则关闭网络通道,并返回NC_MSG_UNKNOWN表示未知错误;如果获取到的数据可用,则退出循环,返回获取到的数据。
循环的目的是保证能够不断地从网络通道中获取数据,直到获取到数据为止。在循环中,代码会不断地进行读取数据的尝试,直到获取到数据或者等待超时为止。如果循环执行完毕后仍未获取到数据,则说明网络通道中没有可读数据。
阅读全文