forked from eden-emu/eden
		
	hle: service: nvdrv: Implement SyncpointManager, to manage syncpoints.
This commit is contained in:
		
							parent
							
								
									bca9591660
								
							
						
					
					
						commit
						d567b7e841
					
				
					 4 changed files with 127 additions and 1 deletions
				
			
		|  | @ -454,6 +454,8 @@ add_library(core STATIC | |||
|     hle/service/nvdrv/nvdrv.h | ||||
|     hle/service/nvdrv/nvmemp.cpp | ||||
|     hle/service/nvdrv/nvmemp.h | ||||
|     hle/service/nvdrv/syncpoint_manager.cpp | ||||
|     hle/service/nvdrv/syncpoint_manager.h | ||||
|     hle/service/nvflinger/buffer_queue.cpp | ||||
|     hle/service/nvflinger/buffer_queue.h | ||||
|     hle/service/nvflinger/nvflinger.cpp | ||||
|  |  | |||
|  | @ -36,7 +36,7 @@ void InstallInterfaces(SM::ServiceManager& service_manager, NVFlinger::NVFlinger | |||
|     nvflinger.SetNVDrvInstance(module_); | ||||
| } | ||||
| 
 | ||||
| Module::Module(Core::System& system) { | ||||
| Module::Module(Core::System& system) : syncpoint_manager{system.GPU()} { | ||||
|     auto& kernel = system.Kernel(); | ||||
|     for (u32 i = 0; i < MaxNvEvents; i++) { | ||||
|         std::string event_label = fmt::format("NVDRV::NvEvent_{}", i); | ||||
|  |  | |||
							
								
								
									
										39
									
								
								src/core/hle/service/nvdrv/syncpoint_manager.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								src/core/hle/service/nvdrv/syncpoint_manager.cpp
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,39 @@ | |||
| // Copyright 2020 yuzu emulator team
 | ||||
| // Licensed under GPLv2 or any later version
 | ||||
| // Refer to the license.txt file included.
 | ||||
| 
 | ||||
| #include "common/assert.h" | ||||
| #include "core/hle/service/nvdrv/syncpoint_manager.h" | ||||
| #include "video_core/gpu.h" | ||||
| 
 | ||||
| namespace Service::Nvidia { | ||||
| 
 | ||||
| SyncpointManager::SyncpointManager(Tegra::GPU& gpu) : gpu{gpu} {} | ||||
| 
 | ||||
| SyncpointManager::~SyncpointManager() = default; | ||||
| 
 | ||||
| u32 SyncpointManager::RefreshSyncpoint(u32 syncpoint_id) { | ||||
|     syncpoints[syncpoint_id].min = gpu.GetSyncpointValue(syncpoint_id); | ||||
|     return GetSyncpointMin(syncpoint_id); | ||||
| } | ||||
| 
 | ||||
| u32 SyncpointManager::AllocateSyncpoint() { | ||||
|     for (u32 syncpoint_id = 1; syncpoint_id < MaxSyncPoints; syncpoint_id++) { | ||||
|         if (!syncpoints[syncpoint_id].is_allocated) { | ||||
|             syncpoints[syncpoint_id].is_allocated = true; | ||||
|             return syncpoint_id; | ||||
|         } | ||||
|     } | ||||
|     UNREACHABLE_MSG("No more available syncpoints!"); | ||||
|     return {}; | ||||
| } | ||||
| 
 | ||||
| u32 SyncpointManager::IncreaseSyncpoint(u32 syncpoint_id, u32 value) { | ||||
|     for (u32 index = 0; index < value; ++index) { | ||||
|         syncpoints[syncpoint_id].max.fetch_add(1, std::memory_order_relaxed); | ||||
|     } | ||||
| 
 | ||||
|     return GetSyncpointMax(syncpoint_id); | ||||
| } | ||||
| 
 | ||||
| } // namespace Service::Nvidia
 | ||||
							
								
								
									
										85
									
								
								src/core/hle/service/nvdrv/syncpoint_manager.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										85
									
								
								src/core/hle/service/nvdrv/syncpoint_manager.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,85 @@ | |||
| // Copyright 2020 yuzu emulator team
 | ||||
| // Licensed under GPLv2 or any later version
 | ||||
| // Refer to the license.txt file included.
 | ||||
| 
 | ||||
| #pragma once | ||||
| 
 | ||||
| #include <array> | ||||
| #include <atomic> | ||||
| 
 | ||||
| #include "common/common_types.h" | ||||
| #include "core/hle/service/nvdrv/nvdata.h" | ||||
| 
 | ||||
| namespace Tegra { | ||||
| class GPU; | ||||
| } | ||||
| 
 | ||||
| namespace Service::Nvidia { | ||||
| 
 | ||||
| class SyncpointManager final { | ||||
| public: | ||||
|     explicit SyncpointManager(Tegra::GPU& gpu); | ||||
|     ~SyncpointManager(); | ||||
| 
 | ||||
|     /**
 | ||||
|      * Returns true if the specified syncpoint is expired for the given value. | ||||
|      * @param syncpoint_id Syncpoint ID to check. | ||||
|      * @param value Value to check against the specified syncpoint. | ||||
|      * @returns True if the specified syncpoint is expired for the given value, otherwise False. | ||||
|      */ | ||||
|     bool IsSyncpointExpired(u32 syncpoint_id, u32 value) const { | ||||
|         return (GetSyncpointMax(syncpoint_id) - value) >= (GetSyncpointMin(syncpoint_id) - value); | ||||
|     } | ||||
| 
 | ||||
|     /**
 | ||||
|      * Gets the lower bound for the specified syncpoint. | ||||
|      * @param syncpoint_id Syncpoint ID to get the lower bound for. | ||||
|      * @returns The lower bound for the specified syncpoint. | ||||
|      */ | ||||
|     u32 GetSyncpointMin(u32 syncpoint_id) const { | ||||
|         return syncpoints[syncpoint_id].min.load(std::memory_order_relaxed); | ||||
|     } | ||||
| 
 | ||||
|     /**
 | ||||
|      * Gets the uper bound for the specified syncpoint. | ||||
|      * @param syncpoint_id Syncpoint ID to get the upper bound for. | ||||
|      * @returns The upper bound for the specified syncpoint. | ||||
|      */ | ||||
|     u32 GetSyncpointMax(u32 syncpoint_id) const { | ||||
|         return syncpoints[syncpoint_id].max.load(std::memory_order_relaxed); | ||||
|     } | ||||
| 
 | ||||
|     /**
 | ||||
|      * Refreshes the minimum value for the specified syncpoint. | ||||
|      * @param syncpoint_id Syncpoint ID to be refreshed. | ||||
|      * @returns The new syncpoint minimum value. | ||||
|      */ | ||||
|     u32 RefreshSyncpoint(u32 syncpoint_id); | ||||
| 
 | ||||
|     /**
 | ||||
|      * Allocates a new syncoint. | ||||
|      * @returns The syncpoint ID for the newly allocated syncpoint. | ||||
|      */ | ||||
|     u32 AllocateSyncpoint(); | ||||
| 
 | ||||
|     /**
 | ||||
|      * Increases the maximum value for the specified syncpoint. | ||||
|      * @param syncpoint_id Syncpoint ID to be increased. | ||||
|      * @param value Value to increase the specified syncpoint by. | ||||
|      * @returns The new syncpoint maximum value. | ||||
|      */ | ||||
|     u32 IncreaseSyncpoint(u32 syncpoint_id, u32 value); | ||||
| 
 | ||||
| private: | ||||
|     struct Syncpoint { | ||||
|         std::atomic<u32> min; | ||||
|         std::atomic<u32> max; | ||||
|         std::atomic<bool> is_allocated; | ||||
|     }; | ||||
| 
 | ||||
|     std::array<Syncpoint, MaxSyncPoints> syncpoints{}; | ||||
| 
 | ||||
|     Tegra::GPU& gpu; | ||||
| }; | ||||
| 
 | ||||
| } // namespace Service::Nvidia
 | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 bunnei
						bunnei