nuttx/sched/pthread/pthread_condwait.c
yinshengkai d7f02a8cb6 sched: change pthread_mutex implementation from sem to mutex
Since pthread_mutex is implemented by sem, it is impossible to see in ps who holds the lock and causes the wait.
Replace sem with mutex implementation to solve the above problems

Signed-off-by: yinshengkai <yinshengkai@xiaomi.com>
2024-09-06 09:42:53 +08:00

131 lines
3.4 KiB
C

/****************************************************************************
* sched/pthread/pthread_condwait.c
*
* 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.
*
****************************************************************************/
/****************************************************************************
* Included Files
****************************************************************************/
#include <nuttx/config.h>
#include <unistd.h>
#include <pthread.h>
#include <sched.h>
#include <errno.h>
#include <debug.h>
#include <nuttx/cancelpt.h>
#include "pthread/pthread.h"
/****************************************************************************
* Public Functions
****************************************************************************/
/****************************************************************************
* Name: int pthread_cond_wait
*
* Description:
* A thread can wait for a condition variable to be signalled or broadcast.
*
* Input Parameters:
* None
*
* Returned Value:
* None
*
* Assumptions:
*
****************************************************************************/
int pthread_cond_wait(FAR pthread_cond_t *cond, FAR pthread_mutex_t *mutex)
{
int status;
int ret;
irqstate_t flags;
sinfo("cond=%p mutex=%p\n", cond, mutex);
/* pthread_cond_wait() is a cancellation point */
enter_cancellation_point();
/* Make sure that non-NULL references were provided. */
if (cond == NULL || mutex == NULL)
{
ret = EINVAL;
}
/* Make sure that the caller holds the mutex */
else if (!mutex_is_hold(&mutex->mutex))
{
ret = EPERM;
}
else
{
unsigned int nlocks;
/* Give up the mutex */
sinfo("Give up mutex / take cond\n");
flags = enter_critical_section();
sched_lock();
ret = pthread_mutex_breaklock(mutex, &nlocks);
/* Take the semaphore. This may be awakened only be a signal (EINTR)
* or if the thread is canceled (ECANCELED)
*/
status = pthread_sem_take(&cond->sem, NULL);
if (ret == OK)
{
/* Report the first failure that occurs */
ret = status;
}
sched_unlock();
leave_critical_section(flags);
/* Reacquire the mutex.
*
* When cancellation points are enabled, we need to hold the mutex
* when the pthread is canceled and cleanup handlers, if any, are
* entered.
*/
sinfo("Reacquire mutex...\n");
status = pthread_mutex_restorelock(mutex, nlocks);
if (ret == OK)
{
/* Report the first failure that occurs */
ret = status;
}
}
leave_cancellation_point();
sinfo("Returning %d\n", ret);
return ret;
}