mirror of
				https://git.zaroz.cloud/nintendo-back-up/yuzu/yuzu-mainline.git
				synced 2025-03-21 01:53:15 +00:00 
			
		
		
		
	Common/Fiber: Implement Rewinding.
This commit is contained in:
		
							parent
							
								
									41013381d6
								
							
						
					
					
						commit
						137d862d9b
					
				@ -12,10 +12,13 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
namespace Common {
 | 
					namespace Common {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					constexpr std::size_t default_stack_size = 256 * 1024; // 256kb
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#if defined(_WIN32) || defined(WIN32)
 | 
					#if defined(_WIN32) || defined(WIN32)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
struct Fiber::FiberImpl {
 | 
					struct Fiber::FiberImpl {
 | 
				
			||||||
    LPVOID handle = nullptr;
 | 
					    LPVOID handle = nullptr;
 | 
				
			||||||
 | 
					    LPVOID rewind_handle = nullptr;
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void Fiber::start() {
 | 
					void Fiber::start() {
 | 
				
			||||||
@ -26,15 +29,29 @@ void Fiber::start() {
 | 
				
			|||||||
    UNREACHABLE();
 | 
					    UNREACHABLE();
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void Fiber::onRewind() {
 | 
				
			||||||
 | 
					    ASSERT(impl->handle != nullptr);
 | 
				
			||||||
 | 
					    DeleteFiber(impl->handle);
 | 
				
			||||||
 | 
					    impl->handle = impl->rewind_handle;
 | 
				
			||||||
 | 
					    impl->rewind_handle = nullptr;
 | 
				
			||||||
 | 
					    rewind_point(rewind_parameter);
 | 
				
			||||||
 | 
					    UNREACHABLE();
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void __stdcall Fiber::FiberStartFunc(void* fiber_parameter) {
 | 
					void __stdcall Fiber::FiberStartFunc(void* fiber_parameter) {
 | 
				
			||||||
    auto fiber = static_cast<Fiber*>(fiber_parameter);
 | 
					    auto fiber = static_cast<Fiber*>(fiber_parameter);
 | 
				
			||||||
    fiber->start();
 | 
					    fiber->start();
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void __stdcall Fiber::RewindStartFunc(void* fiber_parameter) {
 | 
				
			||||||
 | 
					    auto fiber = static_cast<Fiber*>(fiber_parameter);
 | 
				
			||||||
 | 
					    fiber->onRewind();
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
Fiber::Fiber(std::function<void(void*)>&& entry_point_func, void* start_parameter)
 | 
					Fiber::Fiber(std::function<void(void*)>&& entry_point_func, void* start_parameter)
 | 
				
			||||||
    : entry_point{std::move(entry_point_func)}, start_parameter{start_parameter} {
 | 
					    : entry_point{std::move(entry_point_func)}, start_parameter{start_parameter} {
 | 
				
			||||||
    impl = std::make_unique<FiberImpl>();
 | 
					    impl = std::make_unique<FiberImpl>();
 | 
				
			||||||
    impl->handle = CreateFiber(0, &FiberStartFunc, this);
 | 
					    impl->handle = CreateFiber(default_stack_size, &FiberStartFunc, this);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
Fiber::Fiber() {
 | 
					Fiber::Fiber() {
 | 
				
			||||||
@ -60,6 +77,18 @@ void Fiber::Exit() {
 | 
				
			|||||||
    guard.unlock();
 | 
					    guard.unlock();
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void Fiber::SetRewindPoint(std::function<void(void*)>&& rewind_func, void* start_parameter) {
 | 
				
			||||||
 | 
					    rewind_point = std::move(rewind_func);
 | 
				
			||||||
 | 
					    rewind_parameter = start_parameter;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void Fiber::Rewind() {
 | 
				
			||||||
 | 
					    ASSERT(rewind_point);
 | 
				
			||||||
 | 
					    ASSERT(impl->rewind_handle == nullptr);
 | 
				
			||||||
 | 
					    impl->rewind_handle = CreateFiber(default_stack_size, &RewindStartFunc, this);
 | 
				
			||||||
 | 
					    SwitchToFiber(impl->rewind_handle);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void Fiber::YieldTo(std::shared_ptr<Fiber> from, std::shared_ptr<Fiber> to) {
 | 
					void Fiber::YieldTo(std::shared_ptr<Fiber> from, std::shared_ptr<Fiber> to) {
 | 
				
			||||||
    ASSERT_MSG(from != nullptr, "Yielding fiber is null!");
 | 
					    ASSERT_MSG(from != nullptr, "Yielding fiber is null!");
 | 
				
			||||||
    ASSERT_MSG(to != nullptr, "Next fiber is null!");
 | 
					    ASSERT_MSG(to != nullptr, "Next fiber is null!");
 | 
				
			||||||
@ -81,7 +110,6 @@ std::shared_ptr<Fiber> Fiber::ThreadToFiber() {
 | 
				
			|||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#else
 | 
					#else
 | 
				
			||||||
constexpr std::size_t default_stack_size = 1024 * 1024; // 1MB
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
struct Fiber::FiberImpl {
 | 
					struct Fiber::FiberImpl {
 | 
				
			||||||
    alignas(64) std::array<u8, default_stack_size> stack;
 | 
					    alignas(64) std::array<u8, default_stack_size> stack;
 | 
				
			||||||
 | 
				
			|||||||
@ -46,6 +46,10 @@ public:
 | 
				
			|||||||
    static void YieldTo(std::shared_ptr<Fiber> from, std::shared_ptr<Fiber> to);
 | 
					    static void YieldTo(std::shared_ptr<Fiber> from, std::shared_ptr<Fiber> to);
 | 
				
			||||||
    static std::shared_ptr<Fiber> ThreadToFiber();
 | 
					    static std::shared_ptr<Fiber> ThreadToFiber();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    void SetRewindPoint(std::function<void(void*)>&& rewind_func, void* start_parameter);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    void Rewind();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    /// Only call from main thread's fiber
 | 
					    /// Only call from main thread's fiber
 | 
				
			||||||
    void Exit();
 | 
					    void Exit();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -58,8 +62,10 @@ private:
 | 
				
			|||||||
    Fiber();
 | 
					    Fiber();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#if defined(_WIN32) || defined(WIN32)
 | 
					#if defined(_WIN32) || defined(WIN32)
 | 
				
			||||||
 | 
					    void onRewind();
 | 
				
			||||||
    void start();
 | 
					    void start();
 | 
				
			||||||
    static void FiberStartFunc(void* fiber_parameter);
 | 
					    static void FiberStartFunc(void* fiber_parameter);
 | 
				
			||||||
 | 
					    static void RewindStartFunc(void* fiber_parameter);
 | 
				
			||||||
#else
 | 
					#else
 | 
				
			||||||
    void start(boost::context::detail::transfer_t& transfer);
 | 
					    void start(boost::context::detail::transfer_t& transfer);
 | 
				
			||||||
    static void FiberStartFunc(boost::context::detail::transfer_t transfer);
 | 
					    static void FiberStartFunc(boost::context::detail::transfer_t transfer);
 | 
				
			||||||
@ -69,6 +75,8 @@ private:
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    SpinLock guard{};
 | 
					    SpinLock guard{};
 | 
				
			||||||
    std::function<void(void*)> entry_point{};
 | 
					    std::function<void(void*)> entry_point{};
 | 
				
			||||||
 | 
					    std::function<void(void*)> rewind_point{};
 | 
				
			||||||
 | 
					    void* rewind_parameter{};
 | 
				
			||||||
    void* start_parameter{};
 | 
					    void* start_parameter{};
 | 
				
			||||||
    std::shared_ptr<Fiber> previous_fiber{};
 | 
					    std::shared_ptr<Fiber> previous_fiber{};
 | 
				
			||||||
    std::unique_ptr<FiberImpl> impl;
 | 
					    std::unique_ptr<FiberImpl> impl;
 | 
				
			||||||
 | 
				
			|||||||
@ -309,4 +309,50 @@ TEST_CASE("Fibers::StartRace", "[common]") {
 | 
				
			|||||||
    REQUIRE(test_control.value3 == 1);
 | 
					    REQUIRE(test_control.value3 == 1);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class TestControl4;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static void WorkControl4(void* control);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class TestControl4 {
 | 
				
			||||||
 | 
					public:
 | 
				
			||||||
 | 
					    TestControl4() {
 | 
				
			||||||
 | 
					        fiber1 = std::make_shared<Fiber>(std::function<void(void*)>{WorkControl4}, this);
 | 
				
			||||||
 | 
					        goal_reached = false;
 | 
				
			||||||
 | 
					        rewinded = false;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    void Execute() {
 | 
				
			||||||
 | 
					        thread_fiber = Fiber::ThreadToFiber();
 | 
				
			||||||
 | 
					        Fiber::YieldTo(thread_fiber, fiber1);
 | 
				
			||||||
 | 
					        thread_fiber->Exit();
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    void DoWork() {
 | 
				
			||||||
 | 
					        fiber1->SetRewindPoint(std::function<void(void*)>{WorkControl4}, this);
 | 
				
			||||||
 | 
					        if (rewinded) {
 | 
				
			||||||
 | 
					            goal_reached = true;
 | 
				
			||||||
 | 
					            Fiber::YieldTo(fiber1, thread_fiber);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        rewinded = true;
 | 
				
			||||||
 | 
					        fiber1->Rewind();
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    std::shared_ptr<Common::Fiber> fiber1;
 | 
				
			||||||
 | 
					    std::shared_ptr<Common::Fiber> thread_fiber;
 | 
				
			||||||
 | 
					    bool goal_reached;
 | 
				
			||||||
 | 
					    bool rewinded;
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static void WorkControl4(void* control) {
 | 
				
			||||||
 | 
					    auto* test_control = static_cast<TestControl4*>(control);
 | 
				
			||||||
 | 
					    test_control->DoWork();
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					TEST_CASE("Fibers::Rewind", "[common]") {
 | 
				
			||||||
 | 
					    TestControl4 test_control{};
 | 
				
			||||||
 | 
					    test_control.Execute();
 | 
				
			||||||
 | 
					    REQUIRE(test_control.goal_reached);
 | 
				
			||||||
 | 
					    REQUIRE(test_control.rewinded);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
} // namespace Common
 | 
					} // namespace Common
 | 
				
			||||||
 | 
				
			|||||||
		Loading…
	
		Reference in New Issue
	
	Block a user