mirror of
https://github.com/Atmosphere-NX/Atmosphere
synced 2025-01-12 15:54:47 +00:00
151 lines
5.8 KiB
C++
151 lines
5.8 KiB
C++
/*
|
|
* Copyright (c) 2018-2020 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 "os_inter_process_event.hpp"
|
|
#include "os_inter_process_event_impl.hpp"
|
|
#include "os_waitable_object_list.hpp"
|
|
|
|
namespace ams::os::impl {
|
|
|
|
namespace {
|
|
|
|
inline void SetupInterProcessEventType(InterProcessEventType *event, Handle read_handle, bool read_handle_managed, Handle write_handle, bool write_handle_managed, EventClearMode clear_mode) {
|
|
/* Set handles. */
|
|
event->readable_handle = read_handle;
|
|
event->is_readable_handle_managed = read_handle_managed;
|
|
event->writable_handle = write_handle;
|
|
event->is_writable_handle_managed = write_handle_managed;
|
|
|
|
/* Set auto clear. */
|
|
event->auto_clear = (clear_mode == EventClearMode_AutoClear);
|
|
|
|
/* Create the waitlist node. */
|
|
new (GetPointer(event->waitable_object_list_storage)) impl::WaitableObjectList;
|
|
|
|
/* Set state. */
|
|
event->state = InterProcessEventType::State_Initialized;
|
|
}
|
|
|
|
}
|
|
|
|
Result CreateInterProcessEvent(InterProcessEventType *event, EventClearMode clear_mode) {
|
|
Handle rh, wh;
|
|
R_TRY(impl::InterProcessEventImpl::Create(std::addressof(wh), std::addressof(rh)));
|
|
|
|
SetupInterProcessEventType(event, rh, true, wh, true, clear_mode);
|
|
return ResultSuccess();
|
|
}
|
|
|
|
void DestroyInterProcessEvent(InterProcessEventType *event) {
|
|
AMS_ASSERT(event->state == InterProcessEventType::State_Initialized);
|
|
|
|
/* Clear the state. */
|
|
event->state = InterProcessEventType::State_NotInitialized;
|
|
|
|
/* Close handles if required. */
|
|
if (event->is_readable_handle_managed) {
|
|
if (event->readable_handle != svc::InvalidHandle) {
|
|
impl::InterProcessEventImpl::Close(event->readable_handle);
|
|
}
|
|
event->is_readable_handle_managed = false;
|
|
}
|
|
|
|
if (event->is_writable_handle_managed) {
|
|
if (event->writable_handle != svc::InvalidHandle) {
|
|
impl::InterProcessEventImpl::Close(event->writable_handle);
|
|
}
|
|
event->is_writable_handle_managed = false;
|
|
}
|
|
|
|
/* Destroy the waitlist. */
|
|
GetReference(event->waitable_object_list_storage).~WaitableObjectList();
|
|
}
|
|
|
|
void AttachInterProcessEvent(InterProcessEventType *event, Handle read_handle, bool read_handle_managed, Handle write_handle, bool write_handle_managed, EventClearMode clear_mode) {
|
|
AMS_ASSERT(read_handle != svc::InvalidHandle || write_handle != svc::InvalidHandle);
|
|
|
|
return SetupInterProcessEventType(event, read_handle, read_handle_managed, write_handle, write_handle_managed, clear_mode);
|
|
}
|
|
|
|
Handle DetachReadableHandleOfInterProcessEvent(InterProcessEventType *event) {
|
|
AMS_ASSERT(event->state == InterProcessEventType::State_Initialized);
|
|
|
|
const Handle handle = event->readable_handle;
|
|
|
|
event->readable_handle = svc::InvalidHandle;
|
|
event->is_readable_handle_managed = false;
|
|
|
|
return handle;
|
|
}
|
|
Handle DetachWritableHandleOfInterProcessEvent(InterProcessEventType *event) {
|
|
AMS_ASSERT(event->state == InterProcessEventType::State_Initialized);
|
|
|
|
const Handle handle = event->writable_handle;
|
|
|
|
event->writable_handle = svc::InvalidHandle;
|
|
event->is_writable_handle_managed = false;
|
|
|
|
return handle;
|
|
}
|
|
|
|
void WaitInterProcessEvent(InterProcessEventType *event) {
|
|
AMS_ASSERT(event->state == InterProcessEventType::State_Initialized);
|
|
|
|
return impl::InterProcessEventImpl::Wait(event->readable_handle, event->auto_clear);
|
|
}
|
|
|
|
bool TryWaitInterProcessEvent(InterProcessEventType *event) {
|
|
AMS_ASSERT(event->state == InterProcessEventType::State_Initialized);
|
|
|
|
return impl::InterProcessEventImpl::TryWait(event->readable_handle, event->auto_clear);
|
|
}
|
|
|
|
bool TimedWaitInterProcessEvent(InterProcessEventType *event, TimeSpan timeout) {
|
|
AMS_ASSERT(event->state == InterProcessEventType::State_Initialized);
|
|
AMS_ASSERT(timeout.GetNanoSeconds() >= 0);
|
|
|
|
return impl::InterProcessEventImpl::TimedWait(event->readable_handle, event->auto_clear, timeout);
|
|
}
|
|
|
|
void SignalInterProcessEvent(InterProcessEventType *event) {
|
|
AMS_ASSERT(event->state != InterProcessEventType::State_NotInitialized);
|
|
|
|
return impl::InterProcessEventImpl::Signal(event->writable_handle);
|
|
}
|
|
|
|
void ClearInterProcessEvent(InterProcessEventType *event) {
|
|
AMS_ASSERT(event->state != InterProcessEventType::State_NotInitialized);
|
|
|
|
auto handle = event->readable_handle;
|
|
if (handle == svc::InvalidHandle) {
|
|
handle = event->writable_handle;
|
|
}
|
|
return impl::InterProcessEventImpl::Clear(handle);
|
|
}
|
|
|
|
Handle GetReadableHandleOfInterProcessEvent(const InterProcessEventType *event) {
|
|
AMS_ASSERT(event->state != InterProcessEventType::State_NotInitialized);
|
|
|
|
return event->readable_handle;
|
|
}
|
|
|
|
Handle GetWritableHandleOfInterProcessEvent(const InterProcessEventType *event) {
|
|
AMS_ASSERT(event->state != InterProcessEventType::State_NotInitialized);
|
|
|
|
return event->writable_handle;
|
|
}
|
|
|
|
}
|