/**************************************************************************** * sched/semaphore/sem_waitirq.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 #include #include #include #include #include #include "semaphore/semaphore.h" /**************************************************************************** * Public Functions ****************************************************************************/ /**************************************************************************** * Name: nxsem_wait_irq * * Description: * This function is called when either: * * 1. A signal is received by a task that is waiting on a semaphore. * According to the POSIX spec, "...the calling thread shall not return * from the call to [nxsem_wait] until it either locks the semaphore or * the call is interrupted by a signal." * 2. From logic associated with sem_timedwait(). This function is called * when the timeout elapses without receiving the semaphore. * * Input Parameters: * wtcb - A pointer to the TCB of the task that is waiting on a * semphaphore, but has received a signal or timeout instead. * errcode - EINTR if the semaphore wait was awakened by a signal; * ETIMEDOUT if awakened by a timeout * * Returned Value: * None * * Assumptions: * ****************************************************************************/ void nxsem_wait_irq(FAR struct tcb_s *wtcb, int errcode) { irqstate_t flags; /* Disable interrupts. This is necessary (unfortunately) because an * interrupt handler may attempt to post the semaphore while we are * doing this. */ flags = enter_critical_section(); /* It is possible that an interrupt/context switch beat us to the punch * and already changed the task's state. */ if (wtcb->task_state == TSTATE_WAIT_SEM) { sem_t *sem = wtcb->waitsem; DEBUGASSERT(sem != NULL && sem->semcount < 0); /* Restore the correct priority of all threads that hold references * to this semaphore. */ nxsem_canceled(wtcb, sem); /* And increment the count on the semaphore. This releases the count * that was taken by sem_post(). This count decremented the semaphore * count to negative and caused the thread to be blocked in the first * place. */ sem->semcount++; /* Indicate that the semaphore wait is over. */ wtcb->waitsem = NULL; /* Mark the errno value for the thread. */ wtcb->errcode = errcode; /* Restart the task. */ up_unblock_task(wtcb); } /* Interrupts may now be enabled. */ leave_critical_section(flags); }