/* * 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 . */ #pragma once #include #include #include namespace ams::kern { class KLightLock { private: std::atomic tag; public: constexpr KLightLock() : tag(0) { /* ... */ } void Lock() { const uintptr_t cur_thread = reinterpret_cast(GetCurrentThreadPointer()); while (true) { uintptr_t old_tag = this->tag.load(std::memory_order_relaxed); while (!this->tag.compare_exchange_weak(old_tag, (old_tag == 0) ? cur_thread : old_tag | 1, std::memory_order_acquire)) { /* ... */ } if ((old_tag == 0) || ((old_tag | 1) == (cur_thread | 1))) { break; } this->LockSlowPath(old_tag | 1, cur_thread); } } void Unlock() { const uintptr_t cur_thread = reinterpret_cast(GetCurrentThreadPointer()); uintptr_t expected = cur_thread; if (!this->tag.compare_exchange_weak(expected, 0, std::memory_order_release)) { this->UnlockSlowPath(cur_thread); } } void LockSlowPath(uintptr_t owner, uintptr_t cur_thread); void UnlockSlowPath(uintptr_t cur_thread); }; class KScopedLightLock { private: KLightLock *lock; public: explicit ALWAYS_INLINE KScopedLightLock(KLightLock *l) : lock(l) { this->lock->Lock(); } ALWAYS_INLINE ~KScopedLightLock() { this->lock->Unlock(); } explicit ALWAYS_INLINE KScopedLightLock(KLightLock &l) : KScopedLightLock(std::addressof(l)) { /* ... */ } }; }