| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  | // Copyright 2020 yuzu Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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-02-05 14:13:16 -04:00
										 |  |  | #ifdef _MSC_VER
 | 
					
						
							|  |  |  | #include <windows.h>
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | #include <boost/context/detail/fcontext.hpp>
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace Common { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-10 13:18:23 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  | #ifdef _MSC_VER
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct Fiber::FiberImpl { | 
					
						
							|  |  |  |     LPVOID handle = nullptr; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  | void Fiber::start() { | 
					
						
							| 
									
										
										
										
											2020-02-05 15:48:20 -04:00
										 |  |  |     ASSERT(previous_fiber != nullptr); | 
					
						
							|  |  |  |     previous_fiber->guard.unlock(); | 
					
						
							|  |  |  |     previous_fiber.reset(); | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  |     entry_point(start_parameter); | 
					
						
							| 
									
										
										
										
											2020-02-05 15:48:20 -04:00
										 |  |  |     UNREACHABLE(); | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  | void __stdcall Fiber::FiberStartFunc(void* fiber_parameter) | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  |    auto fiber = static_cast<Fiber *>(fiber_parameter); | 
					
						
							|  |  |  |    fiber->start(); | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Fiber::Fiber(std::function<void(void*)>&& entry_point_func, void* start_parameter) | 
					
						
							|  |  |  |     : guard{}, entry_point{std::move(entry_point_func)}, start_parameter{start_parameter}, previous_fiber{} { | 
					
						
							|  |  |  |     impl = std::make_unique<FiberImpl>(); | 
					
						
							|  |  |  |     impl->handle = CreateFiber(0, &FiberStartFunc, this); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Fiber::Fiber() : guard{}, entry_point{}, start_parameter{}, previous_fiber{} { | 
					
						
							|  |  |  |     impl = std::make_unique<FiberImpl>(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Fiber::~Fiber() { | 
					
						
							|  |  |  |     // Make sure the Fiber is not being used
 | 
					
						
							| 
									
										
										
										
											2020-02-05 15:48:20 -04:00
										 |  |  |     bool locked = guard.try_lock(); | 
					
						
							|  |  |  |     ASSERT_MSG(locked, "Destroying a fiber that's still running"); | 
					
						
							|  |  |  |     if (locked) { | 
					
						
							|  |  |  |         guard.unlock(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  |     DeleteFiber(impl->handle); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Fiber::Exit() { | 
					
						
							| 
									
										
										
										
											2020-02-05 15:48:20 -04:00
										 |  |  |     ASSERT_MSG(is_thread_fiber, "Exitting non main thread fiber"); | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  |     if (!is_thread_fiber) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     ConvertFiberToThread(); | 
					
						
							|  |  |  |     guard.unlock(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Fiber::YieldTo(std::shared_ptr<Fiber> from, std::shared_ptr<Fiber> to) { | 
					
						
							| 
									
										
										
										
											2020-02-05 15:48:20 -04:00
										 |  |  |     ASSERT_MSG(from != nullptr, "Yielding fiber is null!"); | 
					
						
							|  |  |  |     ASSERT_MSG(to != nullptr, "Next fiber is null!"); | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  |     to->guard.lock(); | 
					
						
							|  |  |  |     to->previous_fiber = from; | 
					
						
							|  |  |  |     SwitchToFiber(to->impl->handle); | 
					
						
							|  |  |  |     auto previous_fiber = from->previous_fiber; | 
					
						
							| 
									
										
										
										
											2020-02-05 15:48:20 -04:00
										 |  |  |     ASSERT(previous_fiber != nullptr); | 
					
						
							|  |  |  |     previous_fiber->guard.unlock(); | 
					
						
							|  |  |  |     previous_fiber.reset(); | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::shared_ptr<Fiber> Fiber::ThreadToFiber() { | 
					
						
							|  |  |  |     std::shared_ptr<Fiber> fiber = std::shared_ptr<Fiber>{new Fiber()}; | 
					
						
							|  |  |  |     fiber->guard.lock(); | 
					
						
							|  |  |  |     fiber->impl->handle = ConvertThreadToFiber(NULL); | 
					
						
							|  |  |  |     fiber->is_thread_fiber = true; | 
					
						
							|  |  |  |     return fiber; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | constexpr std::size_t default_stack_size = 1024 * 1024 * 4; // 4MB
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  | struct alignas(64) Fiber::FiberImpl { | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  |     std::array<u8, default_stack_size> stack; | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  |     boost::context::detail::fcontext_t context; | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  | void Fiber::start(boost::context::detail::transfer_t& transfer) { | 
					
						
							| 
									
										
										
										
											2020-02-05 15:48:20 -04:00
										 |  |  |     ASSERT(previous_fiber != nullptr); | 
					
						
							|  |  |  |     previous_fiber->impl->context = transfer.fctx; | 
					
						
							|  |  |  |     previous_fiber->guard.unlock(); | 
					
						
							|  |  |  |     previous_fiber.reset(); | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  |     entry_point(start_parameter); | 
					
						
							| 
									
										
										
										
											2020-02-05 15:48:20 -04:00
										 |  |  |     UNREACHABLE(); | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  | void Fiber::FiberStartFunc(boost::context::detail::transfer_t transfer) | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  | { | 
					
						
							|  |  |  |    auto fiber = static_cast<Fiber *>(transfer.data); | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  |    fiber->start(transfer); | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Fiber::Fiber(std::function<void(void*)>&& entry_point_func, void* start_parameter) | 
					
						
							|  |  |  |     : guard{}, entry_point{std::move(entry_point_func)}, start_parameter{start_parameter}, previous_fiber{} { | 
					
						
							|  |  |  |     impl = std::make_unique<FiberImpl>(); | 
					
						
							| 
									
										
										
										
											2020-02-10 13:18:23 -04:00
										 |  |  |     impl->context = boost::context::detail::make_fcontext(impl->stack.data(), impl->stack.size(), | 
					
						
							|  |  |  |                                                           FiberStartFunc); | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Fiber::Fiber() : guard{}, entry_point{}, start_parameter{}, previous_fiber{} { | 
					
						
							|  |  |  |     impl = std::make_unique<FiberImpl>(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Fiber::~Fiber() { | 
					
						
							|  |  |  |     // Make sure the Fiber is not being used
 | 
					
						
							| 
									
										
										
										
											2020-02-05 15:48:20 -04:00
										 |  |  |     bool locked = guard.try_lock(); | 
					
						
							|  |  |  |     ASSERT_MSG(locked, "Destroying a fiber that's still running"); | 
					
						
							|  |  |  |     if (locked) { | 
					
						
							|  |  |  |         guard.unlock(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Fiber::Exit() { | 
					
						
							| 
									
										
										
										
											2020-02-05 15:48:20 -04:00
										 |  |  |     ASSERT_MSG(is_thread_fiber, "Exitting non main thread fiber"); | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  |     if (!is_thread_fiber) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     guard.unlock(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Fiber::YieldTo(std::shared_ptr<Fiber> from, std::shared_ptr<Fiber> to) { | 
					
						
							| 
									
										
										
										
											2020-02-05 15:48:20 -04:00
										 |  |  |     ASSERT_MSG(from != nullptr, "Yielding fiber is null!"); | 
					
						
							|  |  |  |     ASSERT_MSG(to != nullptr, "Next fiber is null!"); | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  |     to->guard.lock(); | 
					
						
							|  |  |  |     to->previous_fiber = from; | 
					
						
							| 
									
										
										
										
											2020-02-10 13:18:23 -04:00
										 |  |  |     auto transfer = boost::context::detail::jump_fcontext(to->impl->context, nullptr); | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  |     auto previous_fiber = from->previous_fiber; | 
					
						
							| 
									
										
										
										
											2020-02-05 15:48:20 -04:00
										 |  |  |     ASSERT(previous_fiber != nullptr); | 
					
						
							|  |  |  |     previous_fiber->impl->context = transfer.fctx; | 
					
						
							|  |  |  |     previous_fiber->guard.unlock(); | 
					
						
							|  |  |  |     previous_fiber.reset(); | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::shared_ptr<Fiber> Fiber::ThreadToFiber() { | 
					
						
							|  |  |  |     std::shared_ptr<Fiber> fiber = std::shared_ptr<Fiber>{new Fiber()}; | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  |     fiber->guard.lock(); | 
					
						
							| 
									
										
										
										
											2020-02-04 15:06:23 -04:00
										 |  |  |     fiber->is_thread_fiber = true; | 
					
						
							|  |  |  |     return fiber; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } // namespace Common
 |