mirror of
https://github.com/Atmosphere-NX/Atmosphere
synced 2025-01-21 22:26:10 +00:00
os: implement Barrier
This commit is contained in:
parent
b25218c918
commit
5dc64bc1f7
5 changed files with 236 additions and 0 deletions
|
@ -48,4 +48,5 @@
|
|||
#include <stratosphere/os/os_light_event.hpp>
|
||||
#include <stratosphere/os/os_light_message_queue.hpp>
|
||||
#include <stratosphere/os/os_light_semaphore.hpp>
|
||||
#include <stratosphere/os/os_barrier.hpp>
|
||||
#include <stratosphere/os/os_waitable.hpp>
|
||||
|
|
|
@ -0,0 +1,55 @@
|
|||
/*
|
||||
* Copyright (c) 2018-2020 Atmosphère-NX
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify it
|
||||
* under the terms and conditions of the GNU General Public License,
|
||||
* version 2, as published by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope it will be useful, but WITHOUT
|
||||
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||
* more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#pragma once
|
||||
#include <vapours.hpp>
|
||||
#include <stratosphere/os/os_barrier_types.hpp>
|
||||
#include <stratosphere/os/os_barrier_api.hpp>
|
||||
|
||||
namespace ams::os {
|
||||
|
||||
class Barrier {
|
||||
NON_COPYABLE(Barrier);
|
||||
NON_MOVEABLE(Barrier);
|
||||
private:
|
||||
BarrierType m_barrier;
|
||||
public:
|
||||
explicit Barrier(int num_threads) {
|
||||
InitializeBarrier(std::addressof(m_barrier), num_threads);
|
||||
}
|
||||
|
||||
~Barrier() {
|
||||
FinalizeBarrier(std::addressof(m_barrier));
|
||||
}
|
||||
|
||||
void Await() {
|
||||
return AwaitBarrier(std::addressof(m_barrier));
|
||||
}
|
||||
|
||||
operator BarrierType &() {
|
||||
return m_barrier;
|
||||
}
|
||||
|
||||
operator const BarrierType &() const {
|
||||
return m_barrier;
|
||||
}
|
||||
|
||||
BarrierType *GetBase() {
|
||||
return std::addressof(m_barrier);
|
||||
}
|
||||
};
|
||||
|
||||
}
|
|
@ -0,0 +1,29 @@
|
|||
/*
|
||||
* Copyright (c) 2018-2020 Atmosphère-NX
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify it
|
||||
* under the terms and conditions of the GNU General Public License,
|
||||
* version 2, as published by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope it will be useful, but WITHOUT
|
||||
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||
* more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#pragma once
|
||||
#include <vapours.hpp>
|
||||
|
||||
namespace ams::os {
|
||||
|
||||
struct BarrierType;
|
||||
|
||||
void InitializeBarrier(BarrierType *barrier, int num_threads);
|
||||
void FinalizeBarrier(BarrierType *barrier);
|
||||
|
||||
void AwaitBarrier(BarrierType *barrier);
|
||||
|
||||
}
|
|
@ -0,0 +1,35 @@
|
|||
/*
|
||||
* Copyright (c) 2018-2020 Atmosphère-NX
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify it
|
||||
* under the terms and conditions of the GNU General Public License,
|
||||
* version 2, as published by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope it will be useful, but WITHOUT
|
||||
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||
* more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#pragma once
|
||||
#include <vapours.hpp>
|
||||
#include <stratosphere/os/impl/os_internal_critical_section.hpp>
|
||||
#include <stratosphere/os/impl/os_internal_condition_variable.hpp>
|
||||
|
||||
namespace ams::os {
|
||||
|
||||
struct BarrierType {
|
||||
u16 max_threads;
|
||||
u16 waiting_threads;
|
||||
u32 base_counter_lower;
|
||||
u32 base_counter_upper;
|
||||
|
||||
impl::InternalCriticalSectionStorage cs_barrier;
|
||||
impl::InternalConditionVariableStorage cv_gathered;
|
||||
};
|
||||
static_assert(std::is_trivial<BarrierType>::value);
|
||||
|
||||
}
|
116
libraries/libstratosphere/source/os/os_barrier.cpp
Normal file
116
libraries/libstratosphere/source/os/os_barrier.cpp
Normal file
|
@ -0,0 +1,116 @@
|
|||
/*
|
||||
* Copyright (c) 2018-2020 Atmosphère-NX
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify it
|
||||
* under the terms and conditions of the GNU General Public License,
|
||||
* version 2, as published by the Free Software Foundation.
|
||||
*
|
||||
* This program is distributed in the hope it will be useful, but WITHOUT
|
||||
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||
* more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
#include <stratosphere.hpp>
|
||||
|
||||
namespace ams::os {
|
||||
|
||||
namespace {
|
||||
|
||||
ALWAYS_INLINE bool IsBarrierInitialized(const BarrierType *barrier) {
|
||||
return barrier->max_threads != 0;
|
||||
}
|
||||
|
||||
ALWAYS_INLINE u64 GetBarrierBaseCounterImpl(const BarrierType *barrier) {
|
||||
/* Check pre-conditions. */
|
||||
AMS_ASSERT(util::GetReference(barrier->cs_barrier).IsLockedByCurrentThread());
|
||||
|
||||
/* Convert two u32s to u64. */
|
||||
return (static_cast<u64>(barrier->base_counter_lower) << 0) | (static_cast<u64>(barrier->base_counter_upper) << BITSIZEOF(barrier->base_counter_lower));
|
||||
}
|
||||
|
||||
ALWAYS_INLINE void SetBarrierBaseCounterImpl(BarrierType *barrier, u64 value) {
|
||||
/* Check pre-conditions. */
|
||||
AMS_ASSERT(util::GetReference(barrier->cs_barrier).IsLockedByCurrentThread());
|
||||
|
||||
/* Store as two u32s. */
|
||||
barrier->base_counter_lower = static_cast<u32>(value >> 0);
|
||||
barrier->base_counter_upper = static_cast<u32>(value >> BITSIZEOF(barrier->base_counter_lower));
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
void InitializeBarrier(BarrierType *barrier, int num_threads) {
|
||||
/* Check pre-conditions. */
|
||||
AMS_ASSERT(num_threads >= 1);
|
||||
|
||||
/* Construct objects. */
|
||||
util::ConstructAt(barrier->cs_barrier);
|
||||
util::ConstructAt(barrier->cv_gathered);
|
||||
|
||||
/* Set member variables. */
|
||||
barrier->max_threads = num_threads;
|
||||
barrier->waiting_threads = 0;
|
||||
barrier->base_counter_lower = 0;
|
||||
barrier->base_counter_upper = 0;
|
||||
}
|
||||
|
||||
void FinalizeBarrier(BarrierType *barrier) {
|
||||
/* Check pre-conditions. */
|
||||
AMS_ASSERT(IsBarrierInitialized(barrier));
|
||||
AMS_ASSERT(barrier->waiting_threads == 0);
|
||||
|
||||
/* Clear max threads. */
|
||||
barrier->max_threads = 0;
|
||||
|
||||
/* Destroy objects. */
|
||||
util::DestroyAt(barrier->cs_barrier);
|
||||
util::DestroyAt(barrier->cv_gathered);
|
||||
}
|
||||
|
||||
void AwaitBarrier(BarrierType *barrier) {
|
||||
/* Check pre-conditions. */
|
||||
AMS_ASSERT(IsBarrierInitialized(barrier));
|
||||
|
||||
/* Await the barrier. */
|
||||
{
|
||||
/* Acquire exclusive access to the barrier. */
|
||||
auto &cs = util::GetReference(barrier->cs_barrier);
|
||||
std::scoped_lock lk(cs);
|
||||
|
||||
/* Read barrier state. */
|
||||
const u64 base_counter = GetBarrierBaseCounterImpl(barrier);
|
||||
const auto max_threads = barrier->max_threads;
|
||||
auto waiting_threads = barrier->waiting_threads;
|
||||
|
||||
/* Determine next base counter. */
|
||||
const u64 done_base_counter = base_counter + max_threads;
|
||||
|
||||
/* Increment waiting threads. */
|
||||
++waiting_threads;
|
||||
|
||||
/* Check if all threads have synchronized. */
|
||||
if (waiting_threads >= max_threads) {
|
||||
/* They have, so reset waiting thread count. */
|
||||
barrier->waiting_threads = 0;
|
||||
|
||||
/* Set the updated base counter. */
|
||||
SetBarrierBaseCounterImpl(barrier, done_base_counter);
|
||||
|
||||
/* Broadcast to our cv. */
|
||||
util::GetReference(barrier->cv_gathered).Broadcast();
|
||||
} else {
|
||||
/* More threads are needed, so update waiting thread count. */
|
||||
barrier->waiting_threads = waiting_threads;
|
||||
|
||||
/* Wait for remaining threads to await. */
|
||||
while (GetBarrierBaseCounterImpl(barrier) < done_base_counter) {
|
||||
util::GetReference(barrier->cv_gathered).Wait(std::addressof(cs));
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
}
|
Loading…
Reference in a new issue