mirror of
				https://github.com/Atmosphere-NX/Atmosphere-libs.git
				synced 2025-10-31 03:25:47 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			135 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			135 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) Atmosphère-NX
 | |
|  *
 | |
|  * This program is free software; you can redistribute it and/or modify it
 | |
|  * under the terms and conditions of the GNU General Public License,
 | |
|  * version 2, as published by the Free Software Foundation.
 | |
|  *
 | |
|  * This program is distributed in the hope it will be useful, but WITHOUT
 | |
|  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 | |
|  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for
 | |
|  * more details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU General Public License
 | |
|  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
 | |
|  */
 | |
| #include <stratosphere.hpp>
 | |
| #include "impl/os_multiple_wait_holder_impl.hpp"
 | |
| #include "impl/os_inter_process_event.hpp"
 | |
| #include "impl/os_timeout_helper.hpp"
 | |
| 
 | |
| namespace ams::os {
 | |
| 
 | |
|     Result CreateSystemEvent(SystemEventType *event, EventClearMode clear_mode, bool inter_process) {
 | |
|         if (inter_process) {
 | |
|             R_TRY(impl::CreateInterProcessEvent(std::addressof(event->inter_process_event), clear_mode));
 | |
|             event->state = SystemEventType::State_InitializedAsInterProcessEvent;
 | |
|         } else {
 | |
|             InitializeEvent(std::addressof(event->event), false, clear_mode);
 | |
|             event->state = SystemEventType::State_InitializedAsEvent;
 | |
|         }
 | |
|         R_SUCCEED();
 | |
|     }
 | |
| 
 | |
|     void DestroySystemEvent(SystemEventType *event) {
 | |
|         auto state = event->state;
 | |
|         event->state = SystemEventType::State_NotInitialized;
 | |
| 
 | |
|         switch (state) {
 | |
|             case SystemEventType::State_InitializedAsInterProcessEvent: impl::DestroyInterProcessEvent(std::addressof(event->inter_process_event)); break;
 | |
|             case SystemEventType::State_InitializedAsEvent:             FinalizeEvent(std::addressof(event->event)); break;
 | |
|             AMS_UNREACHABLE_DEFAULT_CASE();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     void AttachSystemEvent(SystemEventType *event, NativeHandle read_handle, bool read_handle_managed, NativeHandle write_handle, bool write_handle_managed, EventClearMode clear_mode) {
 | |
|         AMS_ASSERT(read_handle != os::InvalidNativeHandle || write_handle != os::InvalidNativeHandle);
 | |
|         impl::AttachInterProcessEvent(std::addressof(event->inter_process_event), read_handle, read_handle_managed, write_handle, write_handle_managed, clear_mode);
 | |
|         event->state = SystemEventType::State_InitializedAsInterProcessEvent;
 | |
|     }
 | |
| 
 | |
|     void AttachReadableHandleToSystemEvent(SystemEventType *event, NativeHandle read_handle, bool manage_read_handle, EventClearMode clear_mode) {
 | |
|         return AttachSystemEvent(event, read_handle, manage_read_handle, os::InvalidNativeHandle, false, clear_mode);
 | |
|     }
 | |
| 
 | |
|     void AttachWritableHandleToSystemEvent(SystemEventType *event, NativeHandle write_handle, bool manage_write_handle, EventClearMode clear_mode) {
 | |
|         return AttachSystemEvent(event, os::InvalidNativeHandle, false, write_handle, manage_write_handle, clear_mode);
 | |
|     }
 | |
| 
 | |
|     NativeHandle DetachReadableHandleOfSystemEvent(SystemEventType *event) {
 | |
|         AMS_ASSERT(event->state == SystemEventType::State_InitializedAsInterProcessEvent);
 | |
|         return impl::DetachReadableHandleOfInterProcessEvent(std::addressof(event->inter_process_event));
 | |
|     }
 | |
| 
 | |
|     NativeHandle DetachWritableHandleOfSystemEvent(SystemEventType *event) {
 | |
|         AMS_ASSERT(event->state == SystemEventType::State_InitializedAsInterProcessEvent);
 | |
|         return impl::DetachWritableHandleOfInterProcessEvent(std::addressof(event->inter_process_event));
 | |
|     }
 | |
| 
 | |
|     NativeHandle GetReadableHandleOfSystemEvent(const SystemEventType *event) {
 | |
|         AMS_ASSERT(event->state == SystemEventType::State_InitializedAsInterProcessEvent);
 | |
|         return impl::GetReadableHandleOfInterProcessEvent(std::addressof(event->inter_process_event));
 | |
|     }
 | |
| 
 | |
|     NativeHandle GetWritableHandleOfSystemEvent(const SystemEventType *event) {
 | |
|         AMS_ASSERT(event->state == SystemEventType::State_InitializedAsInterProcessEvent);
 | |
|         return impl::GetWritableHandleOfInterProcessEvent(std::addressof(event->inter_process_event));
 | |
| 
 | |
|     }
 | |
| 
 | |
|     void SignalSystemEvent(SystemEventType *event) {
 | |
|         switch (event->state) {
 | |
|             case SystemEventType::State_InitializedAsInterProcessEvent: return impl::SignalInterProcessEvent(std::addressof(event->inter_process_event));
 | |
|             case SystemEventType::State_InitializedAsEvent:             return SignalEvent(std::addressof(event->event));
 | |
|             AMS_UNREACHABLE_DEFAULT_CASE();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     void WaitSystemEvent(SystemEventType *event) {
 | |
|         switch (event->state) {
 | |
|             case SystemEventType::State_InitializedAsInterProcessEvent: return impl::WaitInterProcessEvent(std::addressof(event->inter_process_event));
 | |
|             case SystemEventType::State_InitializedAsEvent:             return WaitEvent(std::addressof(event->event));
 | |
|             AMS_UNREACHABLE_DEFAULT_CASE();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     bool TryWaitSystemEvent(SystemEventType *event) {
 | |
|         switch (event->state) {
 | |
|             case SystemEventType::State_InitializedAsInterProcessEvent: return impl::TryWaitInterProcessEvent(std::addressof(event->inter_process_event));
 | |
|             case SystemEventType::State_InitializedAsEvent:             return TryWaitEvent(std::addressof(event->event));
 | |
|             AMS_UNREACHABLE_DEFAULT_CASE();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     bool TimedWaitSystemEvent(SystemEventType *event, TimeSpan timeout) {
 | |
|         AMS_ASSERT(timeout.GetNanoSeconds() >= 0);
 | |
| 
 | |
|         switch (event->state) {
 | |
|             case SystemEventType::State_InitializedAsInterProcessEvent: return impl::TimedWaitInterProcessEvent(std::addressof(event->inter_process_event), timeout);
 | |
|             case SystemEventType::State_InitializedAsEvent:             return TimedWaitEvent(std::addressof(event->event), timeout);
 | |
|             AMS_UNREACHABLE_DEFAULT_CASE();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     void ClearSystemEvent(SystemEventType *event) {
 | |
|         switch (event->state) {
 | |
|             case SystemEventType::State_InitializedAsInterProcessEvent: return impl::ClearInterProcessEvent(std::addressof(event->inter_process_event));
 | |
|             case SystemEventType::State_InitializedAsEvent:             return ClearEvent(std::addressof(event->event));
 | |
|             AMS_UNREACHABLE_DEFAULT_CASE();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     void InitializeMultiWaitHolder(MultiWaitHolderType *multi_wait_holder, SystemEventType *event) {
 | |
|         switch (event->state) {
 | |
|             case SystemEventType::State_InitializedAsInterProcessEvent:
 | |
|                 util::ConstructAt(GetReference(multi_wait_holder->impl_storage).holder_of_inter_process_event_storage, std::addressof(event->inter_process_event));
 | |
|                 break;
 | |
|             case SystemEventType::State_InitializedAsEvent:
 | |
|                 util::ConstructAt(GetReference(multi_wait_holder->impl_storage).holder_of_event_storage, std::addressof(event->event));
 | |
|                 break;
 | |
|             AMS_UNREACHABLE_DEFAULT_CASE();
 | |
|         }
 | |
|     }
 | |
| 
 | |
| }
 |