blob: 5727ba95632876a2fb03131348f1bde8d2bb73f5 [file] [log] [blame] [edit]
// Copyright 2017 The Fuchsia Authors
//
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file or at
// https://opensource.org/licenses/MIT
#include "kernel/semaphore.h"
#include <zircon/compiler.h>
#include <zircon/errors.h>
#include <zircon/types.h>
#include <kernel/thread_lock.h>
void Semaphore::Post() {
Guard<SpinLock, IrqSave> guard{ThreadLock::Get()};
// Either the number of waiters in the wait queue, or the semaphore count,
// must be 0. It should never be possible for there to be waiters, and a
// positive count.
DEBUG_ASSERT((count_ == 0) || waitq_.IsEmpty());
// If we have no waiters, increment the count. Otherwise, release a waiter.
if (waitq_.IsEmpty()) {
++count_;
} else {
waitq_.WakeOne(true, ZX_OK);
}
}
zx_status_t Semaphore::Wait(const Deadline& deadline) {
Guard<SpinLock, IrqSave> guard{ThreadLock::Get()};
DEBUG_ASSERT((count_ == 0) || waitq_.IsEmpty());
// If the count is positive, simply decrement it and get out.
if (count_ > 0) {
--count_;
return ZX_OK;
}
// Wait in an interruptible state. We will either be woken by a Post
// operation, or by a timeout or signal. Whatever happens, return the reason
// the wait operation ended.
return waitq_.Block(deadline, Interruptible::Yes);
}