| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  | // Copyright 2020 yuzu Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <atomic>
 | 
					
						
							|  |  |  | #include <cstdlib>
 | 
					
						
							|  |  |  | #include <functional>
 | 
					
						
							|  |  |  | #include <memory>
 | 
					
						
							| 
									
										
										
										
											2020-10-28 01:33:57 -03:00
										 |  |  | #include <mutex>
 | 
					
						
							|  |  |  | #include <stdexcept>
 | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  | #include <thread>
 | 
					
						
							|  |  |  | #include <unordered_map>
 | 
					
						
							|  |  |  | #include <vector>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <catch2/catch.hpp>
 | 
					
						
							| 
									
										
										
										
											2020-10-28 01:33:57 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  | #include "common/common_types.h"
 | 
					
						
							|  |  |  | #include "common/fiber.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Common { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-28 01:33:57 -03:00
										 |  |  | class ThreadIds { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     void Register(u32 id) { | 
					
						
							|  |  |  |         const auto thread_id = std::this_thread::get_id(); | 
					
						
							|  |  |  |         std::scoped_lock lock{mutex}; | 
					
						
							|  |  |  |         if (ids.contains(thread_id)) { | 
					
						
							|  |  |  |             throw std::logic_error{"Registering the same thread twice"}; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         ids.emplace(thread_id, id); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     [[nodiscard]] u32 Get() const { | 
					
						
							|  |  |  |         std::scoped_lock lock{mutex}; | 
					
						
							|  |  |  |         return ids.at(std::this_thread::get_id()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     mutable std::mutex mutex; | 
					
						
							|  |  |  |     std::unordered_map<std::thread::id, u32> ids; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  | class TestControl1 { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     TestControl1() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void DoWork(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void ExecuteThread(u32 id); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-28 01:33:57 -03:00
										 |  |  |     ThreadIds thread_ids; | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  |     std::vector<std::shared_ptr<Common::Fiber>> thread_fibers; | 
					
						
							|  |  |  |     std::vector<std::shared_ptr<Common::Fiber>> work_fibers; | 
					
						
							|  |  |  |     std::vector<u32> items; | 
					
						
							|  |  |  |     std::vector<u32> results; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void WorkControl1(void* control) { | 
					
						
							| 
									
										
										
										
											2020-02-10 14:45:08 -04:00
										 |  |  |     auto* test_control = static_cast<TestControl1*>(control); | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  |     test_control->DoWork(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TestControl1::DoWork() { | 
					
						
							| 
									
										
										
										
											2020-10-28 01:33:57 -03:00
										 |  |  |     const u32 id = thread_ids.Get(); | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  |     u32 value = items[id]; | 
					
						
							|  |  |  |     for (u32 i = 0; i < id; i++) { | 
					
						
							|  |  |  |         value++; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     results[id] = value; | 
					
						
							| 
									
										
										
										
											2021-03-07 13:46:53 -08:00
										 |  |  |     Fiber::YieldTo(work_fibers[id], *thread_fibers[id]); | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TestControl1::ExecuteThread(u32 id) { | 
					
						
							| 
									
										
										
										
											2020-10-28 01:33:57 -03:00
										 |  |  |     thread_ids.Register(id); | 
					
						
							| 
									
										
										
										
											2021-03-05 17:08:17 -08:00
										 |  |  |     auto thread_fiber = Fiber::ThreadToFiber(); | 
					
						
							|  |  |  |     thread_fibers[id] = thread_fiber; | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  |     work_fibers[id] = std::make_shared<Fiber>(std::function<void(void*)>{WorkControl1}, this); | 
					
						
							|  |  |  |     items[id] = rand() % 256; | 
					
						
							| 
									
										
										
										
											2021-03-07 13:46:53 -08:00
										 |  |  |     Fiber::YieldTo(thread_fibers[id], *work_fibers[id]); | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  |     thread_fibers[id]->Exit(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void ThreadStart1(u32 id, TestControl1& test_control) { | 
					
						
							|  |  |  |     test_control.ExecuteThread(id); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-05 15:48:20 -04:00
										 |  |  | /** This test checks for fiber setup configuration and validates that fibers are
 | 
					
						
							|  |  |  |  *  doing all the work required. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  | TEST_CASE("Fibers::Setup", "[common]") { | 
					
						
							| 
									
										
										
										
											2020-06-27 18:20:06 -04:00
										 |  |  |     constexpr std::size_t num_threads = 7; | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  |     TestControl1 test_control{}; | 
					
						
							| 
									
										
										
										
											2020-02-10 14:45:08 -04:00
										 |  |  |     test_control.thread_fibers.resize(num_threads); | 
					
						
							|  |  |  |     test_control.work_fibers.resize(num_threads); | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  |     test_control.items.resize(num_threads, 0); | 
					
						
							|  |  |  |     test_control.results.resize(num_threads, 0); | 
					
						
							|  |  |  |     std::vector<std::thread> threads; | 
					
						
							|  |  |  |     for (u32 i = 0; i < num_threads; i++) { | 
					
						
							|  |  |  |         threads.emplace_back(ThreadStart1, i, std::ref(test_control)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     for (u32 i = 0; i < num_threads; i++) { | 
					
						
							|  |  |  |         threads[i].join(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     for (u32 i = 0; i < num_threads; i++) { | 
					
						
							|  |  |  |         REQUIRE(test_control.items[i] + i == test_control.results[i]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class TestControl2 { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     TestControl2() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void DoWork1() { | 
					
						
							|  |  |  |         trap2 = false; | 
					
						
							| 
									
										
										
										
											2020-02-10 13:33:13 -04:00
										 |  |  |         while (trap.load()) | 
					
						
							|  |  |  |             ; | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  |         for (u32 i = 0; i < 12000; i++) { | 
					
						
							|  |  |  |             value1 += i; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-03-07 13:46:53 -08:00
										 |  |  |         Fiber::YieldTo(fiber1, *fiber3); | 
					
						
							| 
									
										
										
										
											2020-10-28 01:33:57 -03:00
										 |  |  |         const u32 id = thread_ids.Get(); | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  |         assert1 = id == 1; | 
					
						
							|  |  |  |         value2 += 5000; | 
					
						
							| 
									
										
										
										
											2021-03-07 13:46:53 -08:00
										 |  |  |         Fiber::YieldTo(fiber1, *thread_fibers[id]); | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void DoWork2() { | 
					
						
							| 
									
										
										
										
											2020-02-10 13:33:13 -04:00
										 |  |  |         while (trap2.load()) | 
					
						
							|  |  |  |             ; | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  |         value2 = 2000; | 
					
						
							|  |  |  |         trap = false; | 
					
						
							| 
									
										
										
										
											2021-03-07 13:46:53 -08:00
										 |  |  |         Fiber::YieldTo(fiber2, *fiber1); | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  |         assert3 = false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void DoWork3() { | 
					
						
							| 
									
										
										
										
											2020-10-28 01:33:57 -03:00
										 |  |  |         const u32 id = thread_ids.Get(); | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  |         assert2 = id == 0; | 
					
						
							|  |  |  |         value1 += 1000; | 
					
						
							| 
									
										
										
										
											2021-03-07 13:46:53 -08:00
										 |  |  |         Fiber::YieldTo(fiber3, *thread_fibers[id]); | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void ExecuteThread(u32 id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void CallFiber1() { | 
					
						
							| 
									
										
										
										
											2020-10-28 01:33:57 -03:00
										 |  |  |         const u32 id = thread_ids.Get(); | 
					
						
							| 
									
										
										
										
											2021-03-07 13:46:53 -08:00
										 |  |  |         Fiber::YieldTo(thread_fibers[id], *fiber1); | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void CallFiber2() { | 
					
						
							| 
									
										
										
										
											2020-10-28 01:33:57 -03:00
										 |  |  |         const u32 id = thread_ids.Get(); | 
					
						
							| 
									
										
										
										
											2021-03-07 13:46:53 -08:00
										 |  |  |         Fiber::YieldTo(thread_fibers[id], *fiber2); | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void Exit(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool assert1{}; | 
					
						
							|  |  |  |     bool assert2{}; | 
					
						
							|  |  |  |     bool assert3{true}; | 
					
						
							|  |  |  |     u32 value1{}; | 
					
						
							|  |  |  |     u32 value2{}; | 
					
						
							|  |  |  |     std::atomic<bool> trap{true}; | 
					
						
							|  |  |  |     std::atomic<bool> trap2{true}; | 
					
						
							| 
									
										
										
										
											2020-10-28 01:33:57 -03:00
										 |  |  |     ThreadIds thread_ids; | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  |     std::vector<std::shared_ptr<Common::Fiber>> thread_fibers; | 
					
						
							|  |  |  |     std::shared_ptr<Common::Fiber> fiber1; | 
					
						
							|  |  |  |     std::shared_ptr<Common::Fiber> fiber2; | 
					
						
							|  |  |  |     std::shared_ptr<Common::Fiber> fiber3; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void WorkControl2_1(void* control) { | 
					
						
							| 
									
										
										
										
											2020-02-10 14:45:08 -04:00
										 |  |  |     auto* test_control = static_cast<TestControl2*>(control); | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  |     test_control->DoWork1(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void WorkControl2_2(void* control) { | 
					
						
							| 
									
										
										
										
											2020-02-10 14:45:08 -04:00
										 |  |  |     auto* test_control = static_cast<TestControl2*>(control); | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  |     test_control->DoWork2(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void WorkControl2_3(void* control) { | 
					
						
							| 
									
										
										
										
											2020-02-10 14:45:08 -04:00
										 |  |  |     auto* test_control = static_cast<TestControl2*>(control); | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  |     test_control->DoWork3(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TestControl2::ExecuteThread(u32 id) { | 
					
						
							| 
									
										
										
										
											2020-10-28 01:33:57 -03:00
										 |  |  |     thread_ids.Register(id); | 
					
						
							| 
									
										
										
										
											2021-03-05 17:08:17 -08:00
										 |  |  |     auto thread_fiber = Fiber::ThreadToFiber(); | 
					
						
							|  |  |  |     thread_fibers[id] = thread_fiber; | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TestControl2::Exit() { | 
					
						
							| 
									
										
										
										
											2020-10-28 01:33:57 -03:00
										 |  |  |     const u32 id = thread_ids.Get(); | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  |     thread_fibers[id]->Exit(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void ThreadStart2_1(u32 id, TestControl2& test_control) { | 
					
						
							|  |  |  |     test_control.ExecuteThread(id); | 
					
						
							|  |  |  |     test_control.CallFiber1(); | 
					
						
							|  |  |  |     test_control.Exit(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void ThreadStart2_2(u32 id, TestControl2& test_control) { | 
					
						
							|  |  |  |     test_control.ExecuteThread(id); | 
					
						
							|  |  |  |     test_control.CallFiber2(); | 
					
						
							|  |  |  |     test_control.Exit(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-05 15:48:20 -04:00
										 |  |  | /** This test checks for fiber thread exchange configuration and validates that fibers are
 | 
					
						
							| 
									
										
										
										
											2021-01-02 09:00:05 -05:00
										 |  |  |  *  that a fiber has been successfully transferred from one thread to another and that the TLS | 
					
						
							| 
									
										
										
										
											2020-02-05 15:48:20 -04:00
										 |  |  |  *  region of the thread is kept while changing fibers. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  | TEST_CASE("Fibers::InterExchange", "[common]") { | 
					
						
							|  |  |  |     TestControl2 test_control{}; | 
					
						
							| 
									
										
										
										
											2020-02-10 14:45:08 -04:00
										 |  |  |     test_control.thread_fibers.resize(2); | 
					
						
							| 
									
										
										
										
											2020-02-10 13:33:13 -04:00
										 |  |  |     test_control.fiber1 = | 
					
						
							|  |  |  |         std::make_shared<Fiber>(std::function<void(void*)>{WorkControl2_1}, &test_control); | 
					
						
							|  |  |  |     test_control.fiber2 = | 
					
						
							|  |  |  |         std::make_shared<Fiber>(std::function<void(void*)>{WorkControl2_2}, &test_control); | 
					
						
							|  |  |  |     test_control.fiber3 = | 
					
						
							|  |  |  |         std::make_shared<Fiber>(std::function<void(void*)>{WorkControl2_3}, &test_control); | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  |     std::thread thread1(ThreadStart2_1, 0, std::ref(test_control)); | 
					
						
							|  |  |  |     std::thread thread2(ThreadStart2_2, 1, std::ref(test_control)); | 
					
						
							|  |  |  |     thread1.join(); | 
					
						
							|  |  |  |     thread2.join(); | 
					
						
							|  |  |  |     REQUIRE(test_control.assert1); | 
					
						
							|  |  |  |     REQUIRE(test_control.assert2); | 
					
						
							|  |  |  |     REQUIRE(test_control.assert3); | 
					
						
							|  |  |  |     REQUIRE(test_control.value2 == 7000); | 
					
						
							|  |  |  |     u32 cal_value = 0; | 
					
						
							|  |  |  |     for (u32 i = 0; i < 12000; i++) { | 
					
						
							|  |  |  |         cal_value += i; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     cal_value += 1000; | 
					
						
							|  |  |  |     REQUIRE(test_control.value1 == cal_value); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-05 15:48:20 -04:00
										 |  |  | class TestControl3 { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     TestControl3() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void DoWork1() { | 
					
						
							|  |  |  |         value1 += 1; | 
					
						
							| 
									
										
										
										
											2021-03-07 13:46:53 -08:00
										 |  |  |         Fiber::YieldTo(fiber1, *fiber2); | 
					
						
							| 
									
										
										
										
											2020-10-28 01:33:57 -03:00
										 |  |  |         const u32 id = thread_ids.Get(); | 
					
						
							| 
									
										
										
										
											2020-02-05 15:48:20 -04:00
										 |  |  |         value3 += 1; | 
					
						
							| 
									
										
										
										
											2021-03-07 13:46:53 -08:00
										 |  |  |         Fiber::YieldTo(fiber1, *thread_fibers[id]); | 
					
						
							| 
									
										
										
										
											2020-02-05 15:48:20 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void DoWork2() { | 
					
						
							|  |  |  |         value2 += 1; | 
					
						
							| 
									
										
										
										
											2020-10-28 01:33:57 -03:00
										 |  |  |         const u32 id = thread_ids.Get(); | 
					
						
							| 
									
										
										
										
											2021-03-07 13:46:53 -08:00
										 |  |  |         Fiber::YieldTo(fiber2, *thread_fibers[id]); | 
					
						
							| 
									
										
										
										
											2020-02-05 15:48:20 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void ExecuteThread(u32 id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void CallFiber1() { | 
					
						
							| 
									
										
										
										
											2020-10-28 01:33:57 -03:00
										 |  |  |         const u32 id = thread_ids.Get(); | 
					
						
							| 
									
										
										
										
											2021-03-07 13:46:53 -08:00
										 |  |  |         Fiber::YieldTo(thread_fibers[id], *fiber1); | 
					
						
							| 
									
										
										
										
											2020-02-05 15:48:20 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void Exit(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     u32 value1{}; | 
					
						
							|  |  |  |     u32 value2{}; | 
					
						
							|  |  |  |     u32 value3{}; | 
					
						
							| 
									
										
										
										
											2020-10-28 01:33:57 -03:00
										 |  |  |     ThreadIds thread_ids; | 
					
						
							| 
									
										
										
										
											2021-03-05 17:08:17 -08:00
										 |  |  |     std::vector<std::shared_ptr<Common::Fiber>> thread_fibers; | 
					
						
							| 
									
										
										
										
											2020-02-05 15:48:20 -04:00
										 |  |  |     std::shared_ptr<Common::Fiber> fiber1; | 
					
						
							|  |  |  |     std::shared_ptr<Common::Fiber> fiber2; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void WorkControl3_1(void* control) { | 
					
						
							| 
									
										
										
										
											2020-02-10 14:45:08 -04:00
										 |  |  |     auto* test_control = static_cast<TestControl3*>(control); | 
					
						
							| 
									
										
										
										
											2020-02-05 15:48:20 -04:00
										 |  |  |     test_control->DoWork1(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void WorkControl3_2(void* control) { | 
					
						
							| 
									
										
										
										
											2020-02-10 14:45:08 -04:00
										 |  |  |     auto* test_control = static_cast<TestControl3*>(control); | 
					
						
							| 
									
										
										
										
											2020-02-05 15:48:20 -04:00
										 |  |  |     test_control->DoWork2(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TestControl3::ExecuteThread(u32 id) { | 
					
						
							| 
									
										
										
										
											2020-10-28 01:33:57 -03:00
										 |  |  |     thread_ids.Register(id); | 
					
						
							| 
									
										
										
										
											2021-03-05 17:08:17 -08:00
										 |  |  |     auto thread_fiber = Fiber::ThreadToFiber(); | 
					
						
							|  |  |  |     thread_fibers[id] = thread_fiber; | 
					
						
							| 
									
										
										
										
											2020-02-05 15:48:20 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TestControl3::Exit() { | 
					
						
							| 
									
										
										
										
											2020-10-28 01:33:57 -03:00
										 |  |  |     const u32 id = thread_ids.Get(); | 
					
						
							| 
									
										
										
										
											2020-02-05 15:48:20 -04:00
										 |  |  |     thread_fibers[id]->Exit(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void ThreadStart3(u32 id, TestControl3& test_control) { | 
					
						
							|  |  |  |     test_control.ExecuteThread(id); | 
					
						
							|  |  |  |     test_control.CallFiber1(); | 
					
						
							|  |  |  |     test_control.Exit(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** This test checks for one two threads racing for starting the same fiber.
 | 
					
						
							| 
									
										
										
										
											2021-01-02 09:00:05 -05:00
										 |  |  |  *  It checks execution occurred in an ordered manner and by no time there were | 
					
						
							| 
									
										
										
										
											2020-02-05 15:48:20 -04:00
										 |  |  |  *  two contexts at the same time. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | TEST_CASE("Fibers::StartRace", "[common]") { | 
					
						
							|  |  |  |     TestControl3 test_control{}; | 
					
						
							| 
									
										
										
										
											2020-02-10 14:45:08 -04:00
										 |  |  |     test_control.thread_fibers.resize(2); | 
					
						
							| 
									
										
										
										
											2020-02-10 13:33:13 -04:00
										 |  |  |     test_control.fiber1 = | 
					
						
							|  |  |  |         std::make_shared<Fiber>(std::function<void(void*)>{WorkControl3_1}, &test_control); | 
					
						
							|  |  |  |     test_control.fiber2 = | 
					
						
							|  |  |  |         std::make_shared<Fiber>(std::function<void(void*)>{WorkControl3_2}, &test_control); | 
					
						
							| 
									
										
										
										
											2020-02-05 15:48:20 -04:00
										 |  |  |     std::thread thread1(ThreadStart3, 0, std::ref(test_control)); | 
					
						
							|  |  |  |     std::thread thread2(ThreadStart3, 1, std::ref(test_control)); | 
					
						
							|  |  |  |     thread1.join(); | 
					
						
							|  |  |  |     thread2.join(); | 
					
						
							|  |  |  |     REQUIRE(test_control.value1 == 1); | 
					
						
							|  |  |  |     REQUIRE(test_control.value2 == 1); | 
					
						
							|  |  |  |     REQUIRE(test_control.value3 == 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-27 16:32:47 -04:00
										 |  |  | 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(); | 
					
						
							| 
									
										
										
										
											2021-03-07 13:46:53 -08:00
										 |  |  |         Fiber::YieldTo(thread_fiber, *fiber1); | 
					
						
							| 
									
										
										
										
											2020-02-27 16:32:47 -04:00
										 |  |  |         thread_fiber->Exit(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void DoWork() { | 
					
						
							|  |  |  |         fiber1->SetRewindPoint(std::function<void(void*)>{WorkControl4}, this); | 
					
						
							|  |  |  |         if (rewinded) { | 
					
						
							|  |  |  |             goal_reached = true; | 
					
						
							| 
									
										
										
										
											2021-03-07 13:46:53 -08:00
										 |  |  |             Fiber::YieldTo(fiber1, *thread_fiber); | 
					
						
							| 
									
										
										
										
											2020-02-27 16:32:47 -04:00
										 |  |  |         } | 
					
						
							|  |  |  |         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); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-05 14:13:16 -04:00
										 |  |  | } // namespace Common
 |