// Copyright 2014 Citra Emulator Project // Licensed under GPLv2 or any later version // Refer to the license.txt file included. #pragma once #include #include #include #include #include #include #include #include #include "common/common_types.h" #include "core/hle/kernel/object.h" #include "core/hle/kernel/thread.h" #include "core/hle/result.h" // Address arbiters are an underlying kernel synchronization object that can be created/used via // supervisor calls (SVCs). They function as sort of a global lock. Typically, games/other CTR // applications use them as an underlying mechanism to implement thread-safe barriers, events, and // semaphores. //////////////////////////////////////////////////////////////////////////////////////////////////// // Kernel namespace namespace Kernel { class Thread; enum class ArbitrationType : u32 { Signal, WaitIfLessThan, DecrementAndWaitIfLessThan, WaitIfLessThanWithTimeout, DecrementAndWaitIfLessThanWithTimeout, }; class AddressArbiter final : public Object, public WakeupCallback { public: explicit AddressArbiter(KernelSystem& kernel); ~AddressArbiter() override; std::string GetTypeName() const override { return "Arbiter"; } std::string GetName() const override { return name; } static constexpr HandleType HANDLE_TYPE = HandleType::AddressArbiter; HandleType GetHandleType() const override { return HANDLE_TYPE; } std::string name; ///< Name of address arbiter object (optional) ResultCode ArbitrateAddress(std::shared_ptr thread, ArbitrationType type, VAddr address, s32 value, u64 nanoseconds); class Callback; private: KernelSystem& kernel; /// Puts the thread to wait on the specified arbitration address under this address arbiter. void WaitThread(std::shared_ptr thread, VAddr wait_address); /// Resume all threads found to be waiting on the address under this address arbiter void ResumeAllThreads(VAddr address); /// Resume one thread found to be waiting on the address under this address arbiter and return /// the resumed thread. std::shared_ptr ResumeHighestPriorityThread(VAddr address); /// Threads waiting for the address arbiter to be signaled. std::vector> waiting_threads; std::shared_ptr timeout_callback; void WakeUp(ThreadWakeupReason reason, std::shared_ptr thread, std::shared_ptr object); class DummyCallback : public WakeupCallback { public: void WakeUp(ThreadWakeupReason reason, std::shared_ptr thread, std::shared_ptr object) override {} }; friend class boost::serialization::access; template void serialize(Archive& ar, const unsigned int file_version) { ar& boost::serialization::base_object(*this); if (file_version == 1) { // This rigmarole is needed because in past versions, AddressArbiter inherited // WakeupCallback But it turns out this breaks shared_from_this, so we split it out. // Using a dummy class to deserialize a base_object allows compatibility to be // maintained. DummyCallback x; ar& boost::serialization::base_object(x); } ar& name; ar& waiting_threads; if (file_version > 1) { ar& timeout_callback; } } }; } // namespace Kernel BOOST_CLASS_EXPORT_KEY(Kernel::AddressArbiter) BOOST_CLASS_EXPORT_KEY(Kernel::AddressArbiter::Callback) BOOST_CLASS_VERSION(Kernel::AddressArbiter, 2) CONSTRUCT_KERNEL_OBJECT(Kernel::AddressArbiter)