| 
									
										
										
										
											2022-04-23 04:59:50 -04:00
										 |  |  | // SPDX-FileCopyrightText: Copyright 2020 yuzu Emulator Project
 | 
					
						
							|  |  |  | // SPDX-License-Identifier: GPL-2.0-or-later
 | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-07 16:01:26 -07:00
										 |  |  | #include <mutex>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-05 15:48:20 -04:00
										 |  |  | #include "common/assert.h"
 | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  | #include "common/fiber.h"
 | 
					
						
							| 
									
										
										
										
											2020-11-15 00:36:26 -08:00
										 |  |  | #include "common/virtual_buffer.h"
 | 
					
						
							| 
									
										
										
										
											2020-11-06 20:29:54 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  | #include <boost/context/detail/fcontext.hpp>
 | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace Common { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-10 20:37:56 +00:00
										 |  |  | constexpr std::size_t default_stack_size = 512 * 1024; | 
					
						
							| 
									
										
										
										
											2020-02-27 16:32:47 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  | struct Fiber::FiberImpl { | 
					
						
							| 
									
										
										
										
											2020-11-15 00:36:26 -08:00
										 |  |  |     FiberImpl() : stack{default_stack_size}, rewind_stack{default_stack_size} {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     VirtualBuffer<u8> stack; | 
					
						
							|  |  |  |     VirtualBuffer<u8> rewind_stack; | 
					
						
							| 
									
										
										
										
											2020-11-13 15:17:47 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-07 16:01:26 -07:00
										 |  |  |     std::mutex guard; | 
					
						
							| 
									
										
										
										
											2022-07-02 12:33:49 -04:00
										 |  |  |     std::function<void()> entry_point; | 
					
						
							|  |  |  |     std::function<void()> rewind_point; | 
					
						
							| 
									
										
										
										
											2021-03-05 17:08:17 -08:00
										 |  |  |     std::shared_ptr<Fiber> previous_fiber; | 
					
						
							| 
									
										
										
										
											2020-11-06 20:29:54 -05:00
										 |  |  |     bool is_thread_fiber{}; | 
					
						
							|  |  |  |     bool released{}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-13 15:17:47 -08:00
										 |  |  |     u8* stack_limit{}; | 
					
						
							|  |  |  |     u8* rewind_stack_limit{}; | 
					
						
							|  |  |  |     boost::context::detail::fcontext_t context{}; | 
					
						
							|  |  |  |     boost::context::detail::fcontext_t rewind_context{}; | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-02 12:33:49 -04:00
										 |  |  | void Fiber::SetRewindPoint(std::function<void()>&& rewind_func) { | 
					
						
							| 
									
										
										
										
											2020-11-06 20:29:54 -05:00
										 |  |  |     impl->rewind_point = std::move(rewind_func); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-13 13:49:36 -04:00
										 |  |  | void Fiber::Start(boost::context::detail::transfer_t& transfer) { | 
					
						
							| 
									
										
										
										
											2020-11-06 20:29:54 -05:00
										 |  |  |     ASSERT(impl->previous_fiber != nullptr); | 
					
						
							|  |  |  |     impl->previous_fiber->impl->context = transfer.fctx; | 
					
						
							|  |  |  |     impl->previous_fiber->impl->guard.unlock(); | 
					
						
							| 
									
										
										
										
											2021-03-05 17:08:17 -08:00
										 |  |  |     impl->previous_fiber.reset(); | 
					
						
							| 
									
										
										
										
											2022-07-02 12:33:49 -04:00
										 |  |  |     impl->entry_point(); | 
					
						
							| 
									
										
										
										
											2020-02-05 15:48:20 -04:00
										 |  |  |     UNREACHABLE(); | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-13 13:49:36 -04:00
										 |  |  | void Fiber::OnRewind([[maybe_unused]] boost::context::detail::transfer_t& transfer) { | 
					
						
							| 
									
										
										
										
											2020-04-01 09:19:10 -04:00
										 |  |  |     ASSERT(impl->context != nullptr); | 
					
						
							|  |  |  |     impl->context = impl->rewind_context; | 
					
						
							|  |  |  |     impl->rewind_context = nullptr; | 
					
						
							|  |  |  |     u8* tmp = impl->stack_limit; | 
					
						
							|  |  |  |     impl->stack_limit = impl->rewind_stack_limit; | 
					
						
							|  |  |  |     impl->rewind_stack_limit = tmp; | 
					
						
							| 
									
										
										
										
											2022-07-02 12:33:49 -04:00
										 |  |  |     impl->rewind_point(); | 
					
						
							| 
									
										
										
										
											2020-04-01 09:19:10 -04:00
										 |  |  |     UNREACHABLE(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-10 13:33:13 -04:00
										 |  |  | void Fiber::FiberStartFunc(boost::context::detail::transfer_t transfer) { | 
					
						
							| 
									
										
										
										
											2020-11-06 20:29:54 -05:00
										 |  |  |     auto* fiber = static_cast<Fiber*>(transfer.data); | 
					
						
							| 
									
										
										
										
											2020-05-13 13:49:36 -04:00
										 |  |  |     fiber->Start(transfer); | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:19:10 -04:00
										 |  |  | void Fiber::RewindStartFunc(boost::context::detail::transfer_t transfer) { | 
					
						
							| 
									
										
										
										
											2020-11-06 20:29:54 -05:00
										 |  |  |     auto* fiber = static_cast<Fiber*>(transfer.data); | 
					
						
							| 
									
										
										
										
											2020-05-13 13:49:36 -04:00
										 |  |  |     fiber->OnRewind(transfer); | 
					
						
							| 
									
										
										
										
											2020-04-01 09:19:10 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-02 12:33:49 -04:00
										 |  |  | Fiber::Fiber(std::function<void()>&& entry_point_func) : impl{std::make_unique<FiberImpl>()} { | 
					
						
							| 
									
										
										
										
											2020-11-06 20:29:54 -05:00
										 |  |  |     impl->entry_point = std::move(entry_point_func); | 
					
						
							| 
									
										
										
										
											2020-04-01 09:19:10 -04:00
										 |  |  |     impl->stack_limit = impl->stack.data(); | 
					
						
							|  |  |  |     impl->rewind_stack_limit = impl->rewind_stack.data(); | 
					
						
							|  |  |  |     u8* stack_base = impl->stack_limit + default_stack_size; | 
					
						
							| 
									
										
										
										
											2020-02-26 14:39:27 -04:00
										 |  |  |     impl->context = | 
					
						
							| 
									
										
										
										
											2020-02-26 17:34:23 -04:00
										 |  |  |         boost::context::detail::make_fcontext(stack_base, impl->stack.size(), FiberStartFunc); | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-27 18:20:06 -04:00
										 |  |  | Fiber::Fiber() : impl{std::make_unique<FiberImpl>()} {} | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | Fiber::~Fiber() { | 
					
						
							| 
									
										
										
										
											2020-11-06 20:29:54 -05:00
										 |  |  |     if (impl->released) { | 
					
						
							| 
									
										
										
										
											2020-05-13 13:49:36 -04:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  |     // Make sure the Fiber is not being used
 | 
					
						
							| 
									
										
										
										
											2020-11-06 20:29:54 -05:00
										 |  |  |     const bool locked = impl->guard.try_lock(); | 
					
						
							| 
									
										
										
										
											2020-02-05 15:48:20 -04:00
										 |  |  |     ASSERT_MSG(locked, "Destroying a fiber that's still running"); | 
					
						
							|  |  |  |     if (locked) { | 
					
						
							| 
									
										
										
										
											2020-11-06 20:29:54 -05:00
										 |  |  |         impl->guard.unlock(); | 
					
						
							| 
									
										
										
										
											2020-02-05 15:48:20 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Fiber::Exit() { | 
					
						
							| 
									
										
										
										
											2023-03-11 22:10:38 -05:00
										 |  |  |     ASSERT_MSG(impl->is_thread_fiber, "Exiting non main thread fiber"); | 
					
						
							| 
									
										
										
										
											2020-11-06 20:29:54 -05:00
										 |  |  |     if (!impl->is_thread_fiber) { | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-11-06 20:29:54 -05:00
										 |  |  |     impl->guard.unlock(); | 
					
						
							|  |  |  |     impl->released = true; | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 09:19:10 -04:00
										 |  |  | void Fiber::Rewind() { | 
					
						
							| 
									
										
										
										
											2020-11-06 20:29:54 -05:00
										 |  |  |     ASSERT(impl->rewind_point); | 
					
						
							| 
									
										
										
										
											2020-04-01 09:19:10 -04:00
										 |  |  |     ASSERT(impl->rewind_context == nullptr); | 
					
						
							|  |  |  |     u8* stack_base = impl->rewind_stack_limit + default_stack_size; | 
					
						
							|  |  |  |     impl->rewind_context = | 
					
						
							|  |  |  |         boost::context::detail::make_fcontext(stack_base, impl->stack.size(), RewindStartFunc); | 
					
						
							|  |  |  |     boost::context::detail::jump_fcontext(impl->rewind_context, this); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-07 13:46:53 -08:00
										 |  |  | void Fiber::YieldTo(std::weak_ptr<Fiber> weak_from, Fiber& to) { | 
					
						
							|  |  |  |     to.impl->guard.lock(); | 
					
						
							|  |  |  |     to.impl->previous_fiber = weak_from.lock(); | 
					
						
							| 
									
										
										
										
											2021-03-05 22:10:03 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-07 13:46:53 -08:00
										 |  |  |     auto transfer = boost::context::detail::jump_fcontext(to.impl->context, &to); | 
					
						
							| 
									
										
										
										
											2021-03-05 22:10:03 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // "from" might no longer be valid if the thread was killed
 | 
					
						
							| 
									
										
										
										
											2021-03-07 13:46:53 -08:00
										 |  |  |     if (auto from = weak_from.lock()) { | 
					
						
							| 
									
										
										
										
											2022-01-14 16:17:19 -08:00
										 |  |  |         if (from->impl->previous_fiber == nullptr) { | 
					
						
							|  |  |  |             ASSERT_MSG(false, "previous_fiber is nullptr!"); | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-03-05 22:10:03 -08:00
										 |  |  |         from->impl->previous_fiber->impl->context = transfer.fctx; | 
					
						
							|  |  |  |         from->impl->previous_fiber->impl->guard.unlock(); | 
					
						
							|  |  |  |         from->impl->previous_fiber.reset(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-05 17:08:17 -08:00
										 |  |  | std::shared_ptr<Fiber> Fiber::ThreadToFiber() { | 
					
						
							|  |  |  |     std::shared_ptr<Fiber> fiber = std::shared_ptr<Fiber>{new Fiber()}; | 
					
						
							| 
									
										
										
										
											2020-11-06 20:29:54 -05:00
										 |  |  |     fiber->impl->guard.lock(); | 
					
						
							|  |  |  |     fiber->impl->is_thread_fiber = true; | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  |     return fiber; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace Common
 |